blob: 5cf5035fceccf943fbf0f845d61ba58cef289ba1 [file] [edit]
// Copyright 2024 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package tls
import (
"bytes"
"encoding/hex"
"strings"
"testing"
)
func TestDecodeECHConfigLists(t *testing.T) {
for _, tc := range []struct {
list string
numConfigs int
}{
{"0045fe0d0041590020002092a01233db2218518ccbbbbc24df20686af417b37388de6460e94011974777090004000100010012636c6f7564666c6172652d6563682e636f6d0000", 1},
{"0105badd00050504030201fe0d0066000010004104e62b69e2bf659f97be2f1e0d948a4cd5976bb7a91e0d46fbdda9a91e9ddcba5a01e7d697a80a18f9c3c4a31e56e27c8348db161a1cf51d7ef1942d4bcf7222c1000c000100010001000200010003400e7075626c69632e6578616d706c650000fe0d003d00002000207d661615730214aeee70533366f36a609ead65c0c208e62322346ab5bcd8de1c000411112222400e7075626c69632e6578616d706c650000fe0d004d000020002085bd6a03277c25427b52e269e0c77a8eb524ba1eb3d2f132662d4b0ac6cb7357000c000100010001000200010003400e7075626c69632e6578616d706c650008aaaa000474657374", 3},
} {
b, err := hex.DecodeString(tc.list)
if err != nil {
t.Fatal(err)
}
configs, err := parseECHConfigList(b)
if err != nil {
t.Fatal(err)
}
if len(configs) != tc.numConfigs {
t.Fatalf("unexpected number of configs parsed: got %d want %d", len(configs), tc.numConfigs)
}
}
}
func TestSkipBadConfigs(t *testing.T) {
b, err := hex.DecodeString("00c8badd00050504030201fe0d0029006666000401020304000c000100010001000200010003400e7075626c69632e6578616d706c650000fe0d003d000020002072e8a23b7aef67832bcc89d652e3870a60f88ca684ec65d6eace6b61f136064c000411112222400e7075626c69632e6578616d706c650000fe0d004d00002000200ce95810a81d8023f41e83679bc92701b2acd46c75869f95c72bc61c6b12297c000c000100010001000200010003400e7075626c69632e6578616d706c650008aaaa000474657374")
if err != nil {
t.Fatal(err)
}
configs, err := parseECHConfigList(b)
if err != nil {
t.Fatal(err)
}
config, _, _, _ := pickECHConfig(configs)
if config != nil {
t.Fatal("pickECHConfig picked an invalid config")
}
}
func TestPickECHConfigWithInvalidAEADID(t *testing.T) {
b, err := hex.DecodeString("0045fe0d0041590020002092a01233db2218518ccbbbbc24df20686af417b37388de6460e94011974777090004000100010012636c6f7564666c6172652d6563682e636f6d0000")
if err != nil {
t.Fatal(err)
}
buf := bytes.Replace(b, []byte{0x00, 0x01, 0x00, 0x01}, []byte{0x00, 0x01, 0xFF, 0xFF}, 1)
configs, err := parseECHConfigList(buf)
if err != nil {
t.Fatal(err)
}
if config, _, _, _ := pickECHConfig(configs); config != nil {
t.Fatalf("got %v, want nil", config)
}
}
func TestECHPadding(t *testing.T) {
const maxNameLength = 64
for _, tc := range []struct {
name string
serverName string
}{
{"Short", "a.test"},
{"Medium", strings.Repeat("a", 30) + ".test"},
{"MaxLength", strings.Repeat("a", maxNameLength) + ".test"},
{"NoSNI", ""},
} {
t.Run(tc.name, func(t *testing.T) {
inner := &clientHelloMsg{
vers: VersionTLS13,
random: make([]byte, 32),
serverName: tc.serverName,
cipherSuites: []uint16{TLS_AES_128_GCM_SHA256},
compressionMethods: []uint8{0},
supportedVersions: []uint16{VersionTLS13},
}
encoded, err := encodeInnerClientHello(inner, maxNameLength)
if err != nil {
t.Fatal(err)
}
if len(encoded)%32 != 0 {
t.Errorf("got %d, want multiple of 32", len(encoded))
}
})
}
t.Run("SetSizeReduction", func(t *testing.T) {
sizes := make(map[int]struct{})
for sniLen := 1; sniLen <= maxNameLength; sniLen++ {
inner := &clientHelloMsg{
vers: VersionTLS13,
random: make([]byte, 32),
serverName: strings.Repeat("a", sniLen) + ".test",
cipherSuites: []uint16{TLS_AES_128_GCM_SHA256},
compressionMethods: []uint8{0},
supportedVersions: []uint16{VersionTLS13},
}
encoded, err := encodeInnerClientHello(inner, maxNameLength)
if err != nil {
t.Fatal(err)
}
sizes[len(encoded)] = struct{}{}
}
if len(sizes) > 4 {
t.Errorf("got %d distinct encoded sizes for SNI lengths 1..%d, want <= 4", len(sizes), maxNameLength)
}
})
}