In previous chapter, we received first "expected" UDP packet (STUN Binding Request), then sent a STUN Binding Response as an answer. This means that, the client can start DTLS Handshake process.
DTLS (Datagram Transport Layer Security) for secure handshake, authenticating each other, and crypto key exchange process. DTLS is similar to TLS (Transport Layer Security), DTLS runs over UDP instead of TCP. This project supports only DTLS v1.2.
DTLS Handshake consists of some "flights" between client and server, schematized here as:
------ ------
ClientHello --------> Flight 1
<------- HelloVerifyRequest Flight 2
ClientHello --------> Flight 3
ServerHello \
Certificate* \
ServerKeyExchange* Flight 4
CertificateRequest* /
<-------- ServerHelloDone /
Certificate* \
ClientKeyExchange \
CertificateVerify* Flight 5
[ChangeCipherSpec] /
Finished --------> /
[ChangeCipherSpec] \ Flight 6
<-------- Finished /
Sources:
- Breaking Down the TLS Handshake (Before continuing, it's highly recommended to watch this Youtube playlist to understand TLS Handshake process, ciphers, curves, hashing, algorithms, etc... In videos, they tell about TLS v1.2 and v1.3, but we don't use v1.3).
When a new packet comes in, the "AddBuffer" function in backend/src/agent/udpclientsocket.go looks for which protocol standard this packet to rely on.
In this context, we can check out dtls.IsDtlsPacket function. This function checks:
- Data length (arrayLen) is greater than zero
and - First byte of data (ordinally) is between 20 and 63. This byte represents the DTLS Record Header's ContentType value.
If this buffer part complies with these conditions, we can say "this packet is a DTLS protocol packet", then we can process it with DTLS protocol's methods.
from backend/src/dtls/dtlsmessage.go
func IsDtlsPacket(buf []byte, offset int, arrayLen int) bool {
return arrayLen > 0 && buf[offset] >= 20 && buf[offset] <= 63
}
Here is the console output when the server received an "expected" DTLS ClientHello message.
We determined that this packet is DTLS packet. A DTLS packet consists of a Record Header and Content. If ContentType is "Handshake" (22), packet consists of Record Header, Handshake Header and Handshake Content.
The ClientHello packet we received is a "Handshake" message, so we should discuss the structure of a Handshake packet.
from backend/src/dtls/recordheader.go
type RecordHeader struct {
ContentType ContentType
Version DtlsVersion
Epoch uint16
SequenceNumber [SequenceNumberSize]byte //SequenceNumberSize: 6 (48 bit)
Length uint16
}
You can find:
- Detailed information about attributes of this record in WebRTC for the Curious: Securing - DTLS
- ContentType constants in backend/src/stun/messagetype.go
- DtlsVersion constants in backend/src/stun/messagetype.go
If Epoch is zero, the contents of the packet is in clear text, not encrypted. If not, contents are encrypted with the cipher suite which negotiated before while handshake proccess.
Source: Generic header structure of the DTLS record layer
from backend/src/dtls/handshakeheader.go
type HandshakeHeader struct {
HandshakeType HandshakeType
Length uint24
MessageSequence uint16
FragmentOffset uint24
FragmentLength uint24
}
- You can find HandshakeType constants in backend/src/dtls/handshakeheader.go
- "uint24" is defined in backend/src/dtls/dtlsmessage.go, to represent 24 byte unsigned integer values in handshake data structure.
from backend/src/dtls/dtlsmessage.go
type uint24 [3]byte
- MessageSequence: Message Sequence of Handshake Header (different from RecordHeader's SequenceNumber) should be kept in track incoming and outgoing handshake packets as "ClientSequenceNumber" and "ServerHandshakeSequenceNumber" in backend/src/dtls/handshakecontext.go
- DTLS Fragmentation: Handshake messages can be larger (larger than 1KB), e.g. can transmit a X.509 certificate data. In networking, there is a concept called MTU (Maximum Transmission Unit) to optimize some tradeoffs explained here. These principles require to split a larger packet into smaller pieces. These pieces can be transmitted in an unordered way. So we can track the order of the packets and completeness of them using FragmentOffset and FragmentLength information.
DTLS fragmentation is not supported in this project, so we ignore these fragmentation information.
Source: Header structure for the DTLS handshake protocol
The ClientHello message is the first message of the first flight. The RFC counts flights starting from 1, we count them starting from 0, because we assume the Flight 0 is "waiting for ClientHello" state, Flight 1 is after receiving the first ClientHello. So, you can track the flight numbers this way.
from backend/src/dtls/clienthello.go
type ClientHello struct {
Version DtlsVersion
Random Random
Cookie []byte
SessionID []byte
CipherSuiteIDs []CipherSuiteID
CompressionMethodIDs []byte
Extensions map[ExtensionType]Extension
}
Click to expand Wireshark capture (Received): DTLS ClientHello (first, without cookie)
Frame 458: 189 bytes on wire (1512 bits), 189 bytes captured (1512 bits) on interface lo0, id 0
Null/Loopback
Internet Protocol Version 4, Src: 192.168.***.***, Dst: 192.168.***.***
User Datagram Protocol, Src Port: 52993, Dst Port: 15000
Datagram Transport Layer Security
DTLSv1.2 Record Layer: Handshake Protocol: Client Hello
Content Type: Handshake (22)
Version: DTLS 1.0 (0xfeff)
Epoch: 0
Sequence Number: 0
Length: 144
Handshake Protocol: Client Hello
Handshake Type: Client Hello (1)
Length: 132
Message Sequence: 0
Fragment Offset: 0
Fragment Length: 132
Version: DTLS 1.2 (0xfefd)
Random: 78aa217a2ab532e840bf2f989d8f13ec4b66671e1cf1d8627eddd4917e51b31b
GMT Unix Time: Feb 24, 2034 20:40:10.000000000 +03
Random Bytes: 2ab532e840bf2f989d8f13ec4b66671e1cf1d8627eddd4917e51b31b
Session ID Length: 0
Cookie Length: 0
Cipher Suites Length: 22
Cipher Suites (11 suites)
Cipher Suite: TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 (0xc02b)
Cipher Suite: TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 (0xc02f)
Cipher Suite: TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256 (0xcca9)
Cipher Suite: TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256 (0xcca8)
Cipher Suite: TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA (0xc009)
Cipher Suite: TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA (0xc013)
Cipher Suite: TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA (0xc00a)
Cipher Suite: TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA (0xc014)
Cipher Suite: TLS_RSA_WITH_AES_128_GCM_SHA256 (0x009c)
Cipher Suite: TLS_RSA_WITH_AES_128_CBC_SHA (0x002f)
Cipher Suite: TLS_RSA_WITH_AES_256_CBC_SHA (0x0035)
Compression Methods Length: 1
Compression Methods (1 method)
Compression Method: null (0)
Extensions Length: 68
Extension: extended_master_secret (len=0)
Type: extended_master_secret (23)
Length: 0
Extension: renegotiation_info (len=1)
Type: renegotiation_info (65281)
Length: 1
Renegotiation Info extension
Renegotiation info extension length: 0
Extension: supported_groups (len=8)
Type: supported_groups (10)
Length: 8
Supported Groups List Length: 6
Supported Groups (3 groups)
Supported Group: x25519 (0x001d)
Supported Group: secp256r1 (0x0017)
Supported Group: secp384r1 (0x0018)
Extension: ec_point_formats (len=2)
Type: ec_point_formats (11)
Length: 2
EC point formats Length: 1
Elliptic curves point formats (1)
EC point format: uncompressed (0)
Extension: session_ticket (len=0)
Type: session_ticket (35)
Length: 0
Data (0 bytes)
Extension: signature_algorithms (len=20)
Type: signature_algorithms (13)
Length: 20
Signature Hash Algorithms Length: 18
Signature Hash Algorithms (9 algorithms)
Signature Algorithm: ecdsa_secp256r1_sha256 (0x0403)
Signature Hash Algorithm Hash: SHA256 (4)
Signature Hash Algorithm Signature: ECDSA (3)
Signature Algorithm: rsa_pss_rsae_sha256 (0x0804)
Signature Hash Algorithm Hash: Unknown (8)
Signature Hash Algorithm Signature: SM2 (4)
Signature Algorithm: rsa_pkcs1_sha256 (0x0401)
Signature Hash Algorithm Hash: SHA256 (4)
Signature Hash Algorithm Signature: RSA (1)
Signature Algorithm: ecdsa_secp384r1_sha384 (0x0503)
Signature Hash Algorithm Hash: SHA384 (5)
Signature Hash Algorithm Signature: ECDSA (3)
Signature Algorithm: rsa_pss_rsae_sha384 (0x0805)
Signature Hash Algorithm Hash: Unknown (8)
Signature Hash Algorithm Signature: Unknown (5)
Signature Algorithm: rsa_pkcs1_sha384 (0x0501)
Signature Hash Algorithm Hash: SHA384 (5)
Signature Hash Algorithm Signature: RSA (1)
Signature Algorithm: rsa_pss_rsae_sha512 (0x0806)
Signature Hash Algorithm Hash: Unknown (8)
Signature Hash Algorithm Signature: Unknown (6)
Signature Algorithm: rsa_pkcs1_sha512 (0x0601)
Signature Hash Algorithm Hash: SHA512 (6)
Signature Hash Algorithm Signature: RSA (1)
Signature Algorithm: rsa_pkcs1_sha1 (0x0201)
Signature Hash Algorithm Hash: SHA1 (2)
Signature Hash Algorithm Signature: RSA (1)
Extension: use_srtp (len=9)
Type: use_srtp (14)
Length: 9
SRTP Protection Profiles Length: 6
SRTP Protection Profile: SRTP_AES128_CM_HMAC_SHA1_80 (0x0001)
SRTP Protection Profile: SRTP_AEAD_AES_256_GCM (0x0008)
SRTP Protection Profile: SRTP_AEAD_AES_128_GCM (0x0007)
MKI Length: 0
[JA3 Fullstring: 65277,49195-49199-52393-52392-49161-49171-49162-49172-156-47-53,23-65281-10-11-35-13-14,29-23-24,0]
[JA3: c14667d7da3e6f7a7ab5519ef78c2452]
This message is bootstrapper of a new DTLS Handshake process. Client says to us, "I want to make a DTLS handshake with you, these are my security data to share".
- Version: Same logic and structure with the Version of RecordHeader. Value can differ, for e.g. RecordHeader's version can be 1.0 but ClientHello's version can be 1.2.
- Random: Each part (the client and the server) generates random values for themselves, we call them as "Client Random" and "Server Random". In DTLS v1.2, random byte array consists of 32 bytes. The first 4 bytes are for current system time, remaining 28 bytes are randomly generated.
from backend/src/dtls/random.go
type Random struct {
GMTUnixTime time.Time
RandomBytes [RandomBytesLength]byte
}
Source: Pion WebRTC: DTLS Random
-
Cookie: Randomly generated 20 bytes. But in first flight, first ClientHello's Cookie is empty. The second one will come not empty, we will discuss further.
-
SessionID: We didn't use this data, pass empty array, at least in this project.
-
CipherSuiteIDs: Each party has implemented and supported a set of Cipher Suites, in DTLS and TLS each of them coded with uint16 constant value. With this information, the client informs us "I support these cipher suites, choose one of them which you support too, then we can continue using it."
A "Cipher Suite" is a set of algorithms, usually consists of a key exchange algorithm, a hash algorithm and signature/authentication algorithm. You can find cipher suites and values here, and our only one supported cipher suite constant at backend/src/dtls/ciphersuites.go. -
CompressionMethodIDs: We only support Uncompressed mode with value zero.
-
Extensions: Client can send a list of extension data with the ClientHello message. Some of them which we support:
- UseExtendedMasterSecret: In encryption, we use a "master secret". Each party generates their own "master secret" and don't share it with others. There are some ways to generate it, a standard way and an extended way. We will discuss about this, but if ClientHello contains a UseExtendedMasterSecret extension, it means "the client uses extended master secret generation method, you should use it too".
- SupportedEllipticCurves (Supported Groups): Elliptic curve types that the client supports. We support only X25519. You can find further NamedCurve types here, and our only one supported Curve constant (X25519) at backend/src/dtls/ciphersuites.go.
- SupportedPointFormats: We only will use "Uncompressed" value
- UseSRTP: In an unencrypted way, the media streaming is made with RTP (Real-time Transport Protocol) without any encryption. But WebRTC mandates using an encryption mechanism in transportation. If RTP packets are encrypted and authenticated, we call it SRTP (Secure Real-time Transport Protocol).
This extension contains IDs of SRTP Protection Profiles which the client supports. We support only SRTPProtectionProfile_AEAD_AES_128_GCM. You can find further SRTP Protection profile types here, and our only one supported profile constant (AEAD_AES_128_GCM) at backend/src/dtls/ciphersuites.go.
After receiving the first ClientHello (which doesn't contain a Cookie value), we:- Set our DTLS Handshake Context's (HandshakeContext defined in backend/src/dtls/handshakecontext.go) state as "Connecting"
- Set the context's ProtocolVersion as incoming ClientHello's Version. We should check it, but in this project we assume two parties speak with DTLS v1.2.
- Generate a 20 bytes DTLS Cookie by calling "generateDtlsCookie" function, and set the context's Cookie property.
- Set the context's Flight to "Flight 2"
We generate a HelloVerifyRequest message by calling "createDtlsHelloVerifyRequest" function in backend/src/dtls/handshakemanager.go. We share the Cookie data which we generated previously (in context.Cookie) with the client.
from backend/src/dtls/helloverifyrequest.go
type HelloVerifyRequest struct {
Version DtlsVersion
Cookie []byte
}
Click to expand Wireshark capture (Sent): DTLS HelloVerifyRequest
Frame 483: 80 bytes on wire (640 bits), 80 bytes captured (640 bits) on interface lo0, id 0
Null/Loopback
Internet Protocol Version 4, Src: 192.168.***.***, Dst: 192.168.***.***
User Datagram Protocol, Src Port: 15000, Dst Port: 52993
Datagram Transport Layer Security
DTLSv1.2 Record Layer: Handshake Protocol: Hello Verify Request
Content Type: Handshake (22)
Version: DTLS 1.2 (0xfefd)
Epoch: 0
Sequence Number: 0
Length: 35
Handshake Protocol: Hello Verify Request
Handshake Type: Hello Verify Request (3)
Length: 23
Message Sequence: 0
Fragment Offset: 0
Fragment Length: 23
Version: DTLS 1.2 (0xfefd)
Cookie Length: 20
Cookie: b130316a0e21459cd54d85062f2fb018c4f78be0
Now we are waiting for another (nearly the same) ClientHello message, but with a Cookie that has the same value as our HelloVerifyRequest.
Click to expand Wireshark capture (Received): DTLS ClientHello (second, with cookie)
Frame 484: 209 bytes on wire (1672 bits), 209 bytes captured (1672 bits) on interface lo0, id 0
Null/Loopback
Internet Protocol Version 4, Src: 192.168.***.***, Dst: 192.168.***.***
User Datagram Protocol, Src Port: 52993, Dst Port: 15000
Datagram Transport Layer Security
DTLSv1.2 Record Layer: Handshake Protocol: Client Hello
Content Type: Handshake (22)
Version: DTLS 1.0 (0xfeff)
Epoch: 0
Sequence Number: 1
Length: 164
Handshake Protocol: Client Hello
Handshake Type: Client Hello (1)
Length: 152
Message Sequence: 1
Fragment Offset: 0
Fragment Length: 152
Version: DTLS 1.2 (0xfefd)
Random: 78aa217a2ab532e840bf2f989d8f13ec4b66671e1cf1d8627eddd4917e51b31b
GMT Unix Time: Feb 24, 2034 20:40:10.000000000 +03
Random Bytes: 2ab532e840bf2f989d8f13ec4b66671e1cf1d8627eddd4917e51b31b
Session ID Length: 0
Cookie Length: 20
Cookie: b130316a0e21459cd54d85062f2fb018c4f78be0
Cipher Suites Length: 22
Cipher Suites (11 suites)
Cipher Suite: TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 (0xc02b)
Cipher Suite: TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 (0xc02f)
Cipher Suite: TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256 (0xcca9)
Cipher Suite: TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256 (0xcca8)
Cipher Suite: TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA (0xc009)
Cipher Suite: TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA (0xc013)
Cipher Suite: TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA (0xc00a)
Cipher Suite: TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA (0xc014)
Cipher Suite: TLS_RSA_WITH_AES_128_GCM_SHA256 (0x009c)
Cipher Suite: TLS_RSA_WITH_AES_128_CBC_SHA (0x002f)
Cipher Suite: TLS_RSA_WITH_AES_256_CBC_SHA (0x0035)
Compression Methods Length: 1
Compression Methods (1 method)
Compression Method: null (0)
Extensions Length: 68
Extension: extended_master_secret (len=0)
Type: extended_master_secret (23)
Length: 0
Extension: renegotiation_info (len=1)
Type: renegotiation_info (65281)
Length: 1
Renegotiation Info extension
Renegotiation info extension length: 0
Extension: supported_groups (len=8)
Type: supported_groups (10)
Length: 8
Supported Groups List Length: 6
Supported Groups (3 groups)
Supported Group: x25519 (0x001d)
Supported Group: secp256r1 (0x0017)
Supported Group: secp384r1 (0x0018)
Extension: ec_point_formats (len=2)
Type: ec_point_formats (11)
Length: 2
EC point formats Length: 1
Elliptic curves point formats (1)
EC point format: uncompressed (0)
Extension: session_ticket (len=0)
Type: session_ticket (35)
Length: 0
Data (0 bytes)
Extension: signature_algorithms (len=20)
Type: signature_algorithms (13)
Length: 20
Signature Hash Algorithms Length: 18
Signature Hash Algorithms (9 algorithms)
Signature Algorithm: ecdsa_secp256r1_sha256 (0x0403)
Signature Hash Algorithm Hash: SHA256 (4)
Signature Hash Algorithm Signature: ECDSA (3)
Signature Algorithm: rsa_pss_rsae_sha256 (0x0804)
Signature Hash Algorithm Hash: Unknown (8)
Signature Hash Algorithm Signature: SM2 (4)
Signature Algorithm: rsa_pkcs1_sha256 (0x0401)
Signature Hash Algorithm Hash: SHA256 (4)
Signature Hash Algorithm Signature: RSA (1)
Signature Algorithm: ecdsa_secp384r1_sha384 (0x0503)
Signature Hash Algorithm Hash: SHA384 (5)
Signature Hash Algorithm Signature: ECDSA (3)
Signature Algorithm: rsa_pss_rsae_sha384 (0x0805)
Signature Hash Algorithm Hash: Unknown (8)
Signature Hash Algorithm Signature: Unknown (5)
Signature Algorithm: rsa_pkcs1_sha384 (0x0501)
Signature Hash Algorithm Hash: SHA384 (5)
Signature Hash Algorithm Signature: RSA (1)
Signature Algorithm: rsa_pss_rsae_sha512 (0x0806)
Signature Hash Algorithm Hash: Unknown (8)
Signature Hash Algorithm Signature: Unknown (6)
Signature Algorithm: rsa_pkcs1_sha512 (0x0601)
Signature Hash Algorithm Hash: SHA512 (6)
Signature Hash Algorithm Signature: RSA (1)
Signature Algorithm: rsa_pkcs1_sha1 (0x0201)
Signature Hash Algorithm Hash: SHA1 (2)
Signature Hash Algorithm Signature: RSA (1)
Extension: use_srtp (len=9)
Type: use_srtp (14)
Length: 9
SRTP Protection Profiles Length: 6
SRTP Protection Profile: SRTP_AES128_CM_HMAC_SHA1_80 (0x0001)
SRTP Protection Profile: SRTP_AEAD_AES_256_GCM (0x0008)
SRTP Protection Profile: SRTP_AEAD_AES_128_GCM (0x0007)
MKI Length: 0
[JA3 Fullstring: 65277,49195-49199-52393-52392-49161-49171-49162-49172-156-47-53,23-65281-10-11-35-13-14,29-23-24,0]
[JA3: c14667d7da3e6f7a7ab5519ef78c2452]
We received second ClientHello from the client, with nearl same content but with a Cookie.
We:
- Know we are in Flight 2, we check if the incoming ClientHello's Cookie value and we sent via HelloVerifyRequest (stored in our context object). If it is empty, we should return to Flight 0 state and wait for a new ClientHello message. If not empty, we should compare two values.
- Find a cipher suite ID which mutually supported by each peer, via calling "negotiateOnCipherSuiteIDs" function in backend/src/dtls/handshakemanager.go, then set it to context.CipherSuite. In our example, negotiated on TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 (0xc02b) cipher suite.
- Loop through ClientHello message's extensions. We process the known and supported ones, and ignore unknown ones. Tracking the order at console output below:
- UseSRTP extension found: Find an SRTP Protection Profile ID which mutually supported by each peer, via calling "negotiateOnSRTPProtectionProfiles" function in backend/src/dtls/handshakemanager.go, then set it to context.SRTPProtectionProfile. In our example, negotiated on SRTP_AEAD_AES_128_GCM (0x0007) protection profile.
- UseExtendedMasterSecret extension found: We set context.UseExtendedMasterSecret = true, we will generate the master secret with extended way further.
- SupportedEllipticCurves extension found: Find a Named Curve ID which mutually supported by each peer, via calling "negotiateOnCurves" function in backend/src/dtls/handshakemanager.go, then set it to context.Curve. In our example, negotiated on X25519 (0x001d) named curve.
Now, server chose one from each cryptographic methods, algorithms etc... which client offered as alternatives. The server knows which methods they will use while encrypting, the client will learn further. The server should generate some secrets and keys, for it's side.
We:
-
Set incoming ClientHello.Random to context.ClientRandom
-
Generate a Server Random via calling "Generate" function in backend/src/dtls/random.go and set it to context.ServerRandom.
-
Generate a server private and server public key, by using X25519 curve, via calling "GenerateCurveKeypair" function in backend/src/dtls/crypto.go. These private and public keys are different from and not related with previously generated Server Certificate keys.
- Randomly generate 32 bytes array of private key
- Calculate the public key (32 bytes) by using curve25519.ScalarBaseMult
-
Store these generated private key as context.ServerPrivateKey, public key as context.ServerPublicKey.
-
Generate the ServerKeySignature by calling "GenerateKeySignature" function in backend/src/dtls/crypto.go.
-
Generate a signed_params message (signed_params) which consists concatenating of (total 100 bytes) (checkout signed_params enum), via "generateValueKeyMessage" function in backend/src/dtls/crypto.go:
- Client random (32 bytes)
- Server random (32 bytes)
- ECDH Params (4 bytes)
- [0]: NamedCurve (0x03)
- [1:2]: X25519 (0x001d)
- [3]: 32 (public key length)
- Public key (32 bytes)
-
Calculate hash of the "signed_params message" via "hashAlgorithm.Execute" function in backend/src/dtls/ciphersuites.go by sha256.Sum256, hashing result is 32 bytes
-
Sign the hash value via "privateKeyObj.Sign" by ecdsa privateKey.Sign. Result is 72 bytes.
*Attention Note: We hash a byte array contains the context.ServerPublicKey, then sign the result via ServerCertificate.PrivateKey. Public key was generated via "GenerateCurveKeypair", ServerCertificate.PrivateKey was the ServerCertificate's PrivateKey. -
We set this calculated signature to context.ServerKeySignature
-
from backend/src/dtls/handshakemanager.go
context.ServerKeySignature, err = GenerateKeySignature(
clientRandomBytes,
serverRandomBytes,
context.ServerPublicKey,
context.Curve, //x25519
ServerCertificate.PrivateKey,
context.CipherSuite.HashAlgorithm)
from backend/src/dtls/crypto.go
func GenerateKeySignature(clientRandom []byte, serverRandom []byte, publicKey []byte, curve Curve, privateKey crypto.PrivateKey, hashAlgorithm HashAlgorithm) ([]byte, error) {
msg := generateValueKeyMessage(clientRandom, serverRandom, publicKey, curve)
switch privateKeyObj := privateKey.(type) {
case *ecdsa.PrivateKey:
hashed := hashAlgorithm.Execute(msg) //SHA256 sum
logging.Descf(logging.ProtoCRYPTO, "signed_params values hashed: <u>0x%x</u> (<u>%d</u> bytes)", hashed, len(hashed))
signed, err := privateKeyObj.Sign(rand.Reader, hashed, hashAlgorithm.CryptoHashType()) //crypto.SHA256
logging.Descf(logging.ProtoCRYPTO, "signed_params values signed (result will be called as ServerKeySignature): <u>0x%x</u> (<u>%d</u> bytes)", signed, len(signed))
return signed, err
}
return nil, errors.New("not supported private key type")
}
- Set the context's Flight to "Flight 4"
Now we are ready to send a group of messages: A ServerHello, a Certificate, a ServerKeyExchange, a CertificateRequest, and a ServerHelloDone message. These messages can be sent in same packet (by fragmenting) or one by one. We prefer to send them individually, one by one.
Sources:
We generate a ServerHello message by calling "createDtlsServerHello" function in backend/src/dtls/handshakemanager.go.
from backend/src/dtls/serverhello.go
type ServerHello struct {
Version DtlsVersion
Random Random
SessionID []byte
CipherSuiteID CipherSuiteID
CompressionMethodID byte
Extensions map[ExtensionType]Extension
}
Click to expand Wireshark capture (Sent): DTLS ServerHello
Frame 511: 121 bytes on wire (968 bits), 121 bytes captured (968 bits) on interface lo0, id 0
Null/Loopback
Internet Protocol Version 4, Src: 192.168.***.***, Dst: 192.168.***.***
User Datagram Protocol, Src Port: 15000, Dst Port: 52993
Datagram Transport Layer Security
DTLSv1.2 Record Layer: Handshake Protocol: Server Hello
Content Type: Handshake (22)
Version: DTLS 1.2 (0xfefd)
Epoch: 0
Sequence Number: 1
Length: 76
Handshake Protocol: Server Hello
Handshake Type: Server Hello (2)
Length: 64
Message Sequence: 1
Fragment Offset: 0
Fragment Length: 64
Version: DTLS 1.2 (0xfefd)
Random: 627ac5897ff551e05e567951d2b9258207f9bfaf75084047c1fdcca5faad5b64
GMT Unix Time: May 10, 2022 23:05:29.000000000 +03
Random Bytes: 7ff551e05e567951d2b9258207f9bfaf75084047c1fdcca5faad5b64
Session ID Length: 0
Cipher Suite: TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 (0xc02b)
Compression Method: null (0)
Extensions Length: 24
Extension: renegotiation_info (len=1)
Type: renegotiation_info (65281)
Length: 1
Renegotiation Info extension
Renegotiation info extension length: 0
Extension: use_srtp (len=5)
Type: use_srtp (14)
Length: 5
SRTP Protection Profiles Length: 2
SRTP Protection Profile: SRTP_AEAD_AES_128_GCM (0x0007)
MKI Length: 0
Extension: ec_point_formats (len=2)
Type: ec_point_formats (11)
Length: 2
EC point formats Length: 1
Elliptic curves point formats (1)
EC point format: uncompressed (0)
Extension: extended_master_secret (len=0)
Type: extended_master_secret (23)
Length: 0
[JA3S Fullstring: 65277,49195,65281-14-11-23]
[JA3S: eeb7a12006b679344c81e682d2ae5951]
- Set Version to context.ProtocolVersion
- Set Random to context.ServerRandom
- Set CipherSuiteID to context.CipherSuite.ID
- Add UseExtendedMasterSecret extension if context.UseExtendedMasterSecret is true (ClientHello contains UseExtendedMasterSecret extension)
- Add RenegotiationInfo extension (despite we don't support renegotiation)
- Add UseSRTP extension with context.SRTPProtectionProfile value (SRTPProtectionProfile_AEAD_AES_128_GCM - 0x0007)
- Add SupportedPointFormats extension with PointFormatUncompressed (default, zero) value
ServerHello message was sent.
We generate a Certificate message by calling "createDtlsCertificate" function in backend/src/dtls/handshakemanager.go.
from backend/src/dtls/certificate.go
type Certificate struct {
Certificates [][]byte
}
Click to expand Wireshark capture (Sent): DTLS Certificate
Frame 513: 483 bytes on wire (3864 bits), 483 bytes captured (3864 bits) on interface lo0, id 0
Null/Loopback
Internet Protocol Version 4, Src: 192.168.***.***, Dst: 192.168.***.***
User Datagram Protocol, Src Port: 15000, Dst Port: 52993
Datagram Transport Layer Security
DTLSv1.2 Record Layer: Handshake Protocol: Certificate
Content Type: Handshake (22)
Version: DTLS 1.2 (0xfefd)
Epoch: 0
Sequence Number: 2
Length: 438
Handshake Protocol: Certificate
Handshake Type: Certificate (11)
Length: 426
Message Sequence: 2
Fragment Offset: 0
Fragment Length: 426
Certificates Length: 423
Certificates (423 bytes)
Certificate Length: 420
Certificate: 308201a030820146a0030201020211016d6af47768a9349159deffbcaaec2761300a0608… (id-at-commonName=WebRTC-Nuts-and-Bolts)
signedCertificate
version: v3 (2)
serialNumber: 0x016d6af47768a9349159deffbcaaec2761
signature (ecdsa-with-SHA256)
Algorithm Id: 1.2.840.10045.4.3.2 (ecdsa-with-SHA256)
issuer: rdnSequence (0)
rdnSequence: 1 item (id-at-commonName=WebRTC-Nuts-and-Bolts)
RDNSequence item: 1 item (id-at-commonName=WebRTC-Nuts-and-Bolts)
RelativeDistinguishedName item (id-at-commonName=WebRTC-Nuts-and-Bolts)
Id: 2.5.4.3 (id-at-commonName)
DirectoryString: printableString (1)
printableString: WebRTC-Nuts-and-Bolts
validity
notBefore: utcTime (0)
utcTime: 2022-05-10 20:05:04 (UTC)
notAfter: utcTime (0)
utcTime: 2022-11-06 20:05:04 (UTC)
subject: rdnSequence (0)
rdnSequence: 1 item (id-at-commonName=WebRTC-Nuts-and-Bolts)
RDNSequence item: 1 item (id-at-commonName=WebRTC-Nuts-and-Bolts)
RelativeDistinguishedName item (id-at-commonName=WebRTC-Nuts-and-Bolts)
Id: 2.5.4.3 (id-at-commonName)
DirectoryString: printableString (1)
printableString: WebRTC-Nuts-and-Bolts
subjectPublicKeyInfo
algorithm (id-ecPublicKey)
Algorithm Id: 1.2.840.10045.2.1 (id-ecPublicKey)
ECParameters: namedCurve (1)
namedCurve: 1.2.840.10045.3.1.7 (secp256r1)
Padding: 0
subjectPublicKey: 0470c752a7ef812d4c99e2a776188a3ad2b38de6c794dd124fdce1317c2c0a57eb51a983…
extensions: 4 items
Extension (id-ce-keyUsage)
Extension Id: 2.5.29.15 (id-ce-keyUsage)
critical: True
Padding: 2
KeyUsage: a4
1... .... = digitalSignature: True
.0.. .... = contentCommitment: False
..1. .... = keyEncipherment: True
...0 .... = dataEncipherment: False
.... 0... = keyAgreement: False
.... .1.. = keyCertSign: True
.... ..0. = cRLSign: False
.... ...0 = encipherOnly: False
0... .... = decipherOnly: False
Extension (id-ce-extKeyUsage)
Extension Id: 2.5.29.37 (id-ce-extKeyUsage)
KeyPurposeIDs: 2 items
KeyPurposeId: 1.3.6.1.5.5.7.3.2 (id-kp-clientAuth)
KeyPurposeId: 1.3.6.1.5.5.7.3.1 (id-kp-serverAuth)
Extension (id-ce-basicConstraints)
Extension Id: 2.5.29.19 (id-ce-basicConstraints)
critical: True
BasicConstraintsSyntax
cA: True
Extension (id-ce-subjectKeyIdentifier)
Extension Id: 2.5.29.14 (id-ce-subjectKeyIdentifier)
SubjectKeyIdentifier: ec5a0cc3ea06935d4482027434aae62cca6b8b8a
algorithmIdentifier (ecdsa-with-SHA256)
Algorithm Id: 1.2.840.10045.4.3.2 (ecdsa-with-SHA256)
Padding: 0
encrypted: 304502206b40cd413f21a016d5aa53325b2029e1a5ee3bddc2e49a9dadf97a5040718ff8…
- Set Certificates to ServerCertificate.Certificate
We shared our X.509 Server Certificate data with the client.
Certificate message was sent.
We generate a ServerKeyExchange message by calling "createDtlsServerKeyExchange" function in backend/src/dtls/handshakemanager.go.
from backend/src/dtls/serverkeyexchange.go
type ServerKeyExchange struct {
EllipticCurveType CurveType
NamedCurve Curve
PublicKey []byte
AlgoPair AlgoPair
Signature []byte
}
Click to expand Wireshark capture (Sent): DTLS ServerKeyExchange
Frame 514: 169 bytes on wire (1352 bits), 169 bytes captured (1352 bits) on interface lo0, id 0
Null/Loopback
Internet Protocol Version 4, Src: 192.168.***.***, Dst: 192.168.***.***
User Datagram Protocol, Src Port: 15000, Dst Port: 52993
Datagram Transport Layer Security
DTLSv1.2 Record Layer: Handshake Protocol: Server Key Exchange
Content Type: Handshake (22)
Version: DTLS 1.2 (0xfefd)
Epoch: 0
Sequence Number: 3
Length: 124
Handshake Protocol: Server Key Exchange
Handshake Type: Server Key Exchange (12)
Length: 112
Message Sequence: 3
Fragment Offset: 0
Fragment Length: 112
EC Diffie-Hellman Server Params
Curve Type: named_curve (0x03)
Named Curve: x25519 (0x001d)
Pubkey Length: 32
Pubkey: 845dc25870655b8c2c93129b13a4308c375c74179d0de751852aa02a679e6557
Signature Algorithm: ecdsa_secp256r1_sha256 (0x0403)
Signature Hash Algorithm Hash: SHA256 (4)
Signature Hash Algorithm Signature: ECDSA (3)
Signature Length: 72
Signature: 3046022100b8609b89a48945bee8ccad168c935466c2dcf46c8539bbf0f81ac6641e5db2…
- Set EllipticCurveType to context.CurveType (CurveTypeNamedCurve 0x03)
- Set NamedCurve to context.Curve (CurveX25519 0x001d)
- Set PublicKey to context.ServerPublicKey
- Set AlgoPair to AlgoPair{
HashAlgorithm: context.CipherSuite.HashAlgorithm (HashAlgorithmSHA256 - 4)
SignatureAlgorithm: context.CipherSuite.SignatureAlgorithm (SignatureAlgorithmECDSA - 3)
} - Set Signature to context.ServerKeySignature
ServerKeyExchange message was sent.
We generate a CertificateRequest message by calling "createDtlsCertificateRequest" function in backend/src/dtls/handshakemanager.go.
from backend/src/dtls/certificaterequest.go
type CertificateRequest struct {
CertificateTypes []CertificateType
AlgoPairs []AlgoPair
}
Click to expand Wireshark capture (Sent): DTLS CertificateRequest
Frame 515: 65 bytes on wire (520 bits), 65 bytes captured (520 bits) on interface lo0, id 0
Null/Loopback
Internet Protocol Version 4, Src: 192.168.***.***, Dst: 192.168.***.***
User Datagram Protocol, Src Port: 15000, Dst Port: 52993
Datagram Transport Layer Security
DTLSv1.2 Record Layer: Handshake Protocol: Certificate Request
Content Type: Handshake (22)
Version: DTLS 1.2 (0xfefd)
Epoch: 0
Sequence Number: 4
Length: 20
Handshake Protocol: Certificate Request
Handshake Type: Certificate Request (13)
Length: 8
Message Sequence: 4
Fragment Offset: 0
Fragment Length: 8
Certificate types count: 1
Certificate types (1 type)
Certificate type: ECDSA Sign (64)
Signature Hash Algorithms Length: 2
Signature Hash Algorithms (1 algorithm)
Signature Algorithm: ecdsa_secp256r1_sha256 (0x0403)
Signature Hash Algorithm Hash: SHA256 (4)
Signature Hash Algorithm Signature: ECDSA (3)
Distinguished Names Length: 0
- Set CertificateTypes to [CertificateTypeECDSASign (0x40)]
- Set AlgoPairs to [AlgoPair{
HashAlgorithm: context.CipherSuite.HashAlgorithm (HashAlgorithmSHA256 - 4)
SignatureAlgorithm: context.CipherSuite.SignatureAlgorithm (SignatureAlgorithmECDSA - 3)
}]
We requested for a client certificate which ECDSASign type, generated using hash algorithm SHA256 and signature algorithm ECDSA.
CertificateRequest message was sent.
We generate a ServerHelloDone message by calling "createDtlsServerHelloDone" function in backend/src/dtls/handshakemanager.go.
from backend/src/dtls/serverhellodone.go
type ServerHelloDone struct {
}
Click to expand Wireshark capture (Sent): DTLS ServerHelloDone
Frame 516: 57 bytes on wire (456 bits), 57 bytes captured (456 bits) on interface lo0, id 0
Null/Loopback
Internet Protocol Version 4, Src: 192.168.***.***, Dst: 192.168.***.***
User Datagram Protocol, Src Port: 15000, Dst Port: 52993
Datagram Transport Layer Security
DTLSv1.2 Record Layer: Handshake Protocol: Server Hello Done
Content Type: Handshake (22)
Version: DTLS 1.2 (0xfefd)
Epoch: 0
Sequence Number: 5
Length: 12
Handshake Protocol: Server Hello Done
Handshake Type: Server Hello Done (14)
Length: 0
Message Sequence: 5
Fragment Offset: 0
Fragment Length: 0
The message doesn't carry any data.
ServerHelloDone message was sent.
Now we are waiting for a group of messages: A Certificate, a ClientKeyExchange, a CertificateVerify, a ChangeCipherSpec, a Finished message. For e.g., if our client is Chrome, it will send these messages in one packet.
from backend/src/dtls/certificate.go
type Certificate struct {
Certificates [][]byte
}
Click to expand Wireshark capture (Received): DTLS Certificate (came in combined packet)
Frame 522: 579 bytes on wire (4632 bits), 579 bytes captured (4632 bits) on interface lo0, id 0
Null/Loopback
Internet Protocol Version 4, Src: 192.168.***.***, Dst: 192.168.***.***
User Datagram Protocol, Src Port: 52993, Dst Port: 15000
Datagram Transport Layer Security
DTLSv1.2 Record Layer: Handshake Protocol: Certificate
Content Type: Handshake (22)
Version: DTLS 1.2 (0xfefd)
Epoch: 0
Sequence Number: 2
Length: 300
Handshake Protocol: Certificate
Handshake Type: Certificate (11)
Length: 288
Message Sequence: 2
Fragment Offset: 0
Fragment Length: 288
Certificates Length: 285
Certificates (285 bytes)
Certificate Length: 282
Certificate: 308201163081bda00302010202090087e9356f9f2916ea300a06082a8648ce3d04030230… (id-at-commonName=WebRTC)
signedCertificate
version: v3 (2)
serialNumber: 0x0087e9356f9f2916ea
signature (ecdsa-with-SHA256)
Algorithm Id: 1.2.840.10045.4.3.2 (ecdsa-with-SHA256)
issuer: rdnSequence (0)
rdnSequence: 1 item (id-at-commonName=WebRTC)
RDNSequence item: 1 item (id-at-commonName=WebRTC)
RelativeDistinguishedName item (id-at-commonName=WebRTC)
Id: 2.5.4.3 (id-at-commonName)
DirectoryString: uTF8String (4)
uTF8String: WebRTC
validity
notBefore: utcTime (0)
utcTime: 2022-05-09 20:05:26 (UTC)
notAfter: utcTime (0)
utcTime: 2022-06-09 20:05:26 (UTC)
subject: rdnSequence (0)
rdnSequence: 1 item (id-at-commonName=WebRTC)
RDNSequence item: 1 item (id-at-commonName=WebRTC)
RelativeDistinguishedName item (id-at-commonName=WebRTC)
Id: 2.5.4.3 (id-at-commonName)
DirectoryString: uTF8String (4)
uTF8String: WebRTC
subjectPublicKeyInfo
algorithm (id-ecPublicKey)
Algorithm Id: 1.2.840.10045.2.1 (id-ecPublicKey)
ECParameters: namedCurve (1)
namedCurve: 1.2.840.10045.3.1.7 (secp256r1)
Padding: 0
subjectPublicKey: 0417529a3b477764073e1eade96bf4916c96db64f753d61de4dc7ade47799d8ee5899998…
algorithmIdentifier (ecdsa-with-SHA256)
Algorithm Id: 1.2.840.10045.4.3.2 (ecdsa-with-SHA256)
Padding: 0
encrypted: 3045022100fe8422aaf94444773b82c63b20cd031afa23dd7feb4dcb0b2cf12132f04fd6…
...
The client shared their X.509 Server Certificate data with the server.
- Set context.ClientCertificates to message.Certificates
- Call "GetCertificateFingerprintFromBytes" function in backend/src/dtls/crypto.go to generate the fingerprint hash from the certificate byte array
- Compare the calculated fingerprint hash with context.ExpectedFingerprintHash which came with SDP data previously. So we can ensure that "the sender of SDP via Signaling" and "the sender of these handshake messages" are the same person/machine, not another man in the middle.
from backend/src/dtls/clientkeyexchange.go
type ClientKeyExchange struct {
PublicKey []byte
}
Click to expand Wireshark capture (Received): DTLS ClientKeyExchange (came in combined packet)
Frame 522: 579 bytes on wire (4632 bits), 579 bytes captured (4632 bits) on interface lo0, id 0
Null/Loopback
Internet Protocol Version 4, Src: 192.168.***.***, Dst: 192.168.***.***
User Datagram Protocol, Src Port: 52993, Dst Port: 15000
Datagram Transport Layer Security
...
DTLSv1.2 Record Layer: Handshake Protocol: Client Key Exchange
Content Type: Handshake (22)
Version: DTLS 1.2 (0xfefd)
Epoch: 0
Sequence Number: 3
Length: 45
Handshake Protocol: Client Key Exchange
Handshake Type: Client Key Exchange (16)
Length: 33
Message Sequence: 3
Fragment Offset: 0
Fragment Length: 33
EC Diffie-Hellman Client Params
Pubkey Length: 32
Pubkey: 3377c79bef8cf1d25c9a3f9df5f8e4f5977147016afaa3c090dbfbf5f013007a
...
- Set context.ClientKeyExchangePublic to message.PublicKey
- If context.IsCipherSuiteInitialized is false (our cipher suite is not initialized yet), we are ready to initialize our cipher suite, call "initCipherSuite" function in from backend/src/dtls/handshakemanager.go
We need to generate a master secret. Then using this master secret, client random, and server random, we will initialize our keys. We will use these keys while encrypting/decrypting the SRTP packets further.
We call "GeneratePreMasterSecret" function in backend/src/dtls/crypto.go
from backend/src/dtls/handshakemanager.go
preMasterSecret, err := GeneratePreMasterSecret(context.ClientKeyExchangePublic, context.ServerPrivateKey, context.Curve)
- Generate a byte array using the public key which came by ClientKeyExchange message, and our ServerPrivateKey via curve25519.X25519. We call is as "pre-master key"
*Attention Note: Public key was the ClientKeyExchange message's Public Key, the ServerPrivateKey was generated via "GenerateCurveKeypair".
Sources:
We look to our context.UseExtendedMasterSecret boolean value. Remember that, if ClientHello message has UseExtendedMasterSecret extension, we should generate our master secret in extended way.
If context.UseExtendedMasterSecret is true:
-
Concatenate previous (sent and received) handshake messages as one byte array in a row via "concatHandshakeMessages" function in backend/src/dtls/handshakemanager.go.
Don't forget that, we cached latest received and sent messages by handshake message types in context.HandshakeMessagesReceived and context.HandshakeMessagesSent maps.
Order should be like: _ ClientHello (recv) _ ServerHello (sent) _ Certificate (sent) _ ServerKeyExchange (sent) _ CertificateRequest (sent) _ ServerHelloDone (sent) _ Certificate (recv) _ ClientKeyExchange (recv) -
In the screenshoot below (and also the screenshoot in chapter 5.9), we can see the concatenation result took 1179 bytes.
-
Calculate hash of the "handshakeMessages" byte array via "hashAlgorithm.Execute" function in backend/src/dtls/ciphersuites.go by sha256.Sum256, hashing result is 32 bytes. We call this value as "handshakeHash".
-
Call "GenerateExtendedMasterSecret" function in backend/src/dtls/crypto.go with our preMasterSecret, handshakeHash, context.CipherSuite.HashAlgorithm.
from backend/src/dtls/crypto.go
func GenerateExtendedMasterSecret(preMasterSecret []byte, handshakeHash []byte, hashAlgorithm HashAlgorithm) ([]byte, error) {
seed := append([]byte("extended master secret"), handshakeHash...)
result, err := PHash(preMasterSecret, seed, 48, hashAlgorithm)
if err != nil {
return nil, err
}
logging.Descf(logging.ProtoCRYPTO, "Generated Extended MasterSecret using Pre-Master Secret, Handshake Hash via <u>%s</u>: <u>0x%x</u> (<u>%d bytes</u>)", hashAlgorithm, result, len(result))
return result, nil
}
-
In the "GenerateExtendedMasterSecret" function:
- Create the "seed" by concatenating "extended master secret" string value with our handshakeHash byte array. You can find details here.
- Call the "PHash" function backend/src/dtls/crypto.go to generate 48 bytes Extended Master Secret, which is calculated using our preMasterSecret, seed by sha256.Sum256.
"PHash" function is a Pseudorandom Function (PRF), you can find further information here and here. This function returns a byte array with length the requestedLength parameter you gave. In this case, we gave 48 as requestedLength, it returned an array of 48 bytes.
-
Set the result of "GenerateExtendedMasterSecret" function to context.ServerMasterSecret.
If context.UseExtendedMasterSecret is false:
Note: In current context, our code won't come this state, because (I think) Chrome always sends the UseExtendedMasterSecret extension with ClientHello message. But we can discuss on generating non-extended master secret:
-
Call "GenerateMasterSecret" function in backend/src/dtls/crypto.go with our preMasterSecret, clientRandom, serverRandom, context.CipherSuite.HashAlgorithm.
-
In the "GenerateMasterSecret" function:
- Create the "seed" by concatenating "master secret" string value, clientRandom and serverRandom.
Important note: The order of clientRandom and serverRandom is important, can vary by "being a client" or "being a server" as application. - Call the "PHash" function backend/src/dtls/crypto.go to generate 48 bytes Extended Master Secret, which is calculated using our preMasterSecret, seed by sha256.Sum256.
- Create the "seed" by concatenating "master secret" string value, clientRandom and serverRandom.
-
Set the result of "GenerateMasterSecret" function to context.ServerMasterSecret.
Sources:
- RFC 7627: Transport Layer Security (TLS) Session Hash and Extended Master Secret Extension - The Extended Master Secret
- WebRTC for the Curious: Securing - Pseudorandom Function
- Pseudorandom function family (Wikipedia)
- RFC 4346: The Transport Layer Security (TLS) Protocol Version 1.1 - HMAC and the Pseudorandom Function
- Pion WebRTC DTLS project, PHash function (Github)
GCM is abbreviation for "Galois/Counter Mode" and is a type of block cipher mode of operation . In our project, we only implemented the TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 cipher suite, and it uses AES-GCM authenticated encryption.
So, we call the object that makes encryption/decryption over our DTLS Messages (if a message has encrypted) as GCM.
This struct was defined in "dtls" package specifically to process DTLS messages. We have another "GCM" struct in backend/src/srtp/cryptogcm.go, is different from this struct.
You can find the original code on Pion WebRTC DTLS project, GCM struct (Github).
from backend/src/dtls/cryptogcm.go
type GCM struct {
localGCM, remoteGCM cipher.AEAD
localWriteIV, remoteWriteIV []byte
}
-
We call "InitGCM" function in backend/src/dtls/crypto.go with our context.ServerMasterSecret, clientRandom, serverRandom and context.CipherSuite.
-
As you can see here on Pion WebRTC DTLS project, TLSEcdheEcdsaWithAes128GcmSha256 Init function, the cipher suite which we chose has these constant lengths:
- prfMacLen: 0 bytes
- prfKeyLen: 16 bytes (our keys will be 16 bytes length)
- prfIvLen: 4 bytes (our Initialization Vectors will be 4 bytes length)
Note: These constant length values can vary for different cipher suites.
Due to our chosen suite's MAC length is zero, we didn't include things related with MAC in our code.
-
Call "GenerateEncryptionKeys" function in backend/src/dtls/crypto.go, you can find further information here.
-
Create the "seed" by concatenating "key expansion" string value, serverRandom and clientRandom.
Important note: The order of clientRandom and serverRandom is important, can vary by "being a client" or "being a server" as application. -
Generate a keying material by calling "PHash" function with requestedLength value (2 _ keyLen) + (2 _ ivLen).
Because we want: _ A key for client (clientWriteKey) (16 bytes) _ A key for server (serverWriteKey) (16 bytes) _ An initialization vector for client (clientWriteIV) (4 bytes) _ An initialization vector for server (serverWriteIV) (4 bytes)
So we need (16+16+4+4) = 40 bytes array which is generated by PHash with our masterSecret and seed. Then we extract our key and IV values from these 40 bytes sequentially.
from backend/src/dtls/crypto.go
func GenerateEncryptionKeys(masterSecret []byte, clientRandom []byte, serverRandom []byte, keyLen int, ivLen int, hashAlgorithm HashAlgorithm) (*EncryptionKeys, error) {
logging.Descf(logging.ProtoCRYPTO, "Generating encryption keys with Key Length: <u>%d</u>, IV Length: <u>%d</u> via <u>%s</u>, using Master Secret, Server Random, Client Random...", keyLen, ivLen, hashAlgorithm)
seed := append(append([]byte("key expansion"), serverRandom...), clientRandom...)
keyMaterial, err := PHash(masterSecret, seed, (2*keyLen)+(2*ivLen), hashAlgorithm)
if err != nil {
return nil, err
}
clientWriteKey := keyMaterial[:keyLen]
keyMaterial = keyMaterial[keyLen:]
serverWriteKey := keyMaterial[:keyLen]
keyMaterial = keyMaterial[keyLen:]
clientWriteIV := keyMaterial[:ivLen]
keyMaterial = keyMaterial[ivLen:]
serverWriteIV := keyMaterial[:ivLen]
return &EncryptionKeys{
MasterSecret: masterSecret,
ClientWriteKey: clientWriteKey,
ServerWriteKey: serverWriteKey,
ClientWriteIV: clientWriteIV,
ServerWriteIV: serverWriteIV,
}, nil
}
- You can find generated encryption keys below:
- Now, we have an "EncryptionKeys" object defined in backend/src/dtls/crypto.go.
We are ready to create our GCM object, which contains our ciphers and IVs. We pass our key values to our "NewGCM" function.
Important note: The order of pairs (keys.ServerWriteKey, keys.ServerWriteIV) and (keys.ClientWriteKey, keys.ClientWriteIV) while passing them to "NewGCM" function below is important, can vary by "being a client" or "being a server" as application. - While creating our ciphers, we create new instances by aes.NewCipher and cipher.NewGCM
from backend/src/dtls/crypto.go
gcm, err := NewGCM(keys.ServerWriteKey, keys.ServerWriteIV, keys.ClientWriteKey, keys.ClientWriteIV)
from backend/src/dtls/cryptogcm.go
func NewGCM(localKey, localWriteIV, remoteKey, remoteWriteIV []byte) (*GCM, error) {
localBlock, err := aes.NewCipher(localKey)
...
localGCM, err := cipher.NewGCM(localBlock)
...
remoteBlock, err := aes.NewCipher(remoteKey)
...
remoteGCM, err := cipher.NewGCM(remoteBlock)
...
return &GCM{
localGCM: localGCM,
localWriteIV: localWriteIV,
remoteGCM: remoteGCM,
remoteWriteIV: remoteWriteIV,
}, nil
}
- We set returned GCM object to context.GCM and set context.IsCipherSuiteInitialized as true.
Sources:
- What is the main difference between a key, an IV and a nonce? (Stackoverflow)
- Initialization vector
- Keying material (NIST)
- Secret keying material (NIST)
- Key material (Mozilla)
from backend/src/dtls/certificateverify.go
type CertificateVerify struct {
AlgoPair AlgoPair
Signature []byte
}
Click to expand Wireshark capture (Received): DTLS CertificateVerify (came in combined packet)
Frame 522: 579 bytes on wire (4632 bits), 579 bytes captured (4632 bits) on interface lo0, id 0
Null/Loopback
Internet Protocol Version 4, Src: 192.168.***.***, Dst: 192.168.***.***
User Datagram Protocol, Src Port: 52993, Dst Port: 15000
Datagram Transport Layer Security
...
DTLSv1.2 Record Layer: Handshake Protocol: Certificate Verify
Content Type: Handshake (22)
Version: DTLS 1.2 (0xfefd)
Epoch: 0
Sequence Number: 4
Length: 88
Handshake Protocol: Certificate Verify
Handshake Type: Certificate Verify (15)
Length: 76
Message Sequence: 4
Fragment Offset: 0
Fragment Length: 76
Signature Algorithm: ecdsa_secp256r1_sha256 (0x0403)
Signature Hash Algorithm Hash: SHA256 (4)
Signature Hash Algorithm Signature: ECDSA (3)
Signature length: 72
Signature: 30460221008002b23011ec9910f5af308336cbac1b48b6a011dc170ddbfe77dfb0b1a8a1…
...
-
Compare hash algorithm ID of incoming message.AlgoPair.HashAlgorithm and context.CipherSuite.HashAlgorithm
-
Compare signature algorithm ID of incoming message.AlgoPair.SignatureAlgorithm and context.CipherSuite.SignatureAlgorithm
-
Concatenate previous (sent and received) handshake messages as one byte array in a row via "concatHandshakeMessages" function in backend/src/dtls/handshakemanager.go.
Order should be like: _ ClientHello (recv) _ ServerHello (sent) _ Certificate (sent) _ ServerKeyExchange (sent) _ CertificateRequest (sent) _ ServerHelloDone (sent) _ Certificate (recv) _ ClientKeyExchange (recv) -
Call "VerifyCertificate" function in backend/src/dtls/crypto.go with concenated handshakeMessages, context.CipherSuite.HashAlgorithm, message.Signature, context.ClientCertificates
-
Unmarshall the ECDSA R and S values from the Signature of incoming CertificateVerify message by asn1.Unmarshal
-
Calculate hash of the "handshakeMessages" byte array via "hashAlgorithm.Execute" function in backend/src/dtls/ciphersuites.go by sha256.Sum256, hashing result is 32 bytes. We call this value as "handshakeHash".
-
Check the validity of the X.509 certificate which came with Certificate message from the client, by ecdsa.Verify.
from backend/src/dtls/crypto.go
ecdsa.Verify(clientCertificatePublicKey, hash, ecdsaSign.R, ecdsaSign.S)
from backend/src/dtls/changecipherspec.go
type ChangeCipherSpec struct {
}
Click to expand Wireshark capture (Received): DTLS ChangeCipherSpec (came in combined packet)
Frame 522: 579 bytes on wire (4632 bits), 579 bytes captured (4632 bits) on interface lo0, id 0
Null/Loopback
Internet Protocol Version 4, Src: 192.168.***.***, Dst: 192.168.***.***
User Datagram Protocol, Src Port: 52993, Dst Port: 15000
Datagram Transport Layer Security
...
DTLSv1.2 Record Layer: Change Cipher Spec Protocol: Change Cipher Spec
Content Type: Change Cipher Spec (20)
Version: DTLS 1.2 (0xfefd)
Epoch: 0
Sequence Number: 5
Length: 1
Change Cipher Spec Message
...
This type of message contains only a byte with value 1. We don't do anything for this message.
Important note: Epoch of Record Header were 0 and contents were in clear text (not encrypted) until (and including) this ChangeCipherSpec message. But the messages that will come after this, will have Epoch with 1 and will be encrypted.
from backend/src/dtls/finished.go
type Finished struct {
VerifyData []byte
}
Click to expand Wireshark capture (Received): DTLS Finished (came in combined packet)
Important note: As you can see at the end of the block, latest handshake message (Finished) has Epoch: 1, and content was encrypted. Because of this, we can't see contents of it as clear text in Wireshark.
Frame 522: 579 bytes on wire (4632 bits), 579 bytes captured (4632 bits) on interface lo0, id 0
Null/Loopback
Internet Protocol Version 4, Src: 192.168.***.***, Dst: 192.168.***.***
User Datagram Protocol, Src Port: 52993, Dst Port: 15000
Datagram Transport Layer Security
...
Record Layer
Content Type: Handshake (22)
Version: DTLS 1.2 (0xfefd)
Epoch: 1
Sequence Number: 0
Length: 48
Handshake Protocol
(Encrypted content: 0x0001000000000000e22c7b2609d96a7d3a98c7b80e32ea03868231eb419623ce8af27b5ab97b16df143e8743ab0cb6ba)
Important note: Epoch of Record Header is 1 and, this Finished message is the first message which was encrypted. If we can decode and decrypt contents of this message successfully, then verify the VerifyData successfully, it means that, the handshake process for our side succeeded! After that we will send two other messages, then it will be "completely" finished!
-
Concatenate previous (sent and received) handshake messages as one byte array in a row via "concatHandshakeMessages" function in backend/src/dtls/handshakemanager.go.
Order should be like (attention to CertificateVerify and Finished were addded despite previous concatenations): _ ClientHello (recv) _ ServerHello (sent) _ Certificate (sent) _ ServerKeyExchange (sent) _ CertificateRequest (sent) _ ServerHelloDone (sent) _ Certificate (recv) _ ClientKeyExchange (recv) _ CertificateVerify (recv) _ Finished (recv) -
Call "VerifyFinishedData" function in backend/src/dtls/crypto.go with concenated handshakeMessages, context.ServerMasterSecret, context.CipherSuite.HashAlgorithm
-
Calculate hash of the "handshakeMessages" byte array via "hashAlgorithm.GetFunction" function in backend/src/dtls/ciphersuites.go by "Sum" function of the object created by sha256.New.
-
Create the "seed" by concatenating "server finished" string value with our handshakeHash byte array. You can find details here.
-
Call the "PHash" function backend/src/dtls/crypto.go to generate 12 bytes verify data, which is calculated using our context.ServerMasterSecret, seed by sha256.Sum256. This data will be sent via Finished message further.
-
Set the context's Flight to "Flight 6"
Sources:
We generate a ChangeCipherSpec message by calling "createDtlsChangeCipherSpec" function in backend/src/dtls/handshakemanager.go.
from backend/src/dtls/changecipherspec.go
type ChangeCipherSpec struct {
}
Click to expand Wireshark capture (Sent): DTLS ChangeCipherSpec
Frame 549: 46 bytes on wire (368 bits), 46 bytes captured (368 bits) on interface lo0, id 0
Null/Loopback
Internet Protocol Version 4, Src: 192.168.***.***, Dst: 192.168.***.***
User Datagram Protocol, Src Port: 15000, Dst Port: 52993
Datagram Transport Layer Security
DTLSv1.2 Record Layer: Change Cipher Spec Protocol: Change Cipher Spec
Content Type: Change Cipher Spec (20)
Version: DTLS 1.2 (0xfefd)
Epoch: 0
Sequence Number: 6
Length: 1
Change Cipher Spec Message
This type of message contains only a byte with value 1.
Important note: Epoch of Record Header were 0 and contents were in clear text (not encrypted) until (and including) this ChangeCipherSpec message. But the messages will be sent future, will have Epoch with 1, and will be encrypted.
-
ChangeCipherSpec message was sent.
-
Called "IncreaseServerEpoch" function in backend/src/dtls/handshakecontext.go to increase context.ServerEpoch to 1, and set context.ServerSequenceNumber = 0.
We generate a Finished message by calling "createDtlsFinished" function in backend/src/dtls/handshakemanager.go.
from backend/src/dtls/finished.go
type Finished struct {
VerifyData []byte
}
Click to expand Wireshark capture (Sent): DTLS Finished
Important note: As you can see at the end of the block, the handshake message (Finished) has Epoch: 1, and content was encrypted. Because of this, we can't see contents of it as clear text in Wireshark.
Frame 550: 93 bytes on wire (744 bits), 93 bytes captured (744 bits) on interface lo0, id 0
Null/Loopback
Internet Protocol Version 4, Src: 192.168.***.***, Dst: 192.168.***.***
User Datagram Protocol, Src Port: 15000, Dst Port: 52993
Datagram Transport Layer Security
Record Layer
Content Type: Handshake (22)
Version: DTLS 1.2 (0xfefd)
Epoch: 1
Sequence Number: 0
Length: 48
Handshake Protocol
(Encrypted content: 0x61c244d15307ccaf237e2ef34fdd5bddff3d7d17cf3fcb837a80db65f777a2496c609b03d7a35b772754cc206bb16db2)
- Set VerifyData to calculatedVerifyData (calculated previously by calling "VerifyFinishedData" function)
Important note: We sent our first encrypted message successfully!
- Set our DTLS Handshake Context's (HandshakeContext defined in backend/src/dtls/handshakecontext.go) state as "Connected"