mirror of
https://github.com/refraction-networking/utls.git
synced 2025-04-04 12:37:35 +03:00
feat: Chrome 107 fp with shuffler
- added `HelloChrome_107` (not used by `HelloChrome_Auto`) - added `shuffleExtensions()` to shuffle the order of extensions in a `ClientHelloSpec`
This commit is contained in:
parent
862fe372d9
commit
e656c889a8
2 changed files with 66 additions and 1 deletions
|
@ -183,6 +183,7 @@ var (
|
||||||
HelloChrome_96 = ClientHelloID{helloChrome, "96", nil}
|
HelloChrome_96 = ClientHelloID{helloChrome, "96", nil}
|
||||||
HelloChrome_100 = ClientHelloID{helloChrome, "100", nil}
|
HelloChrome_100 = ClientHelloID{helloChrome, "100", nil}
|
||||||
HelloChrome_102 = ClientHelloID{helloChrome, "102", nil}
|
HelloChrome_102 = ClientHelloID{helloChrome, "102", nil}
|
||||||
|
HelloChrome_107 = ClientHelloID{helloChrome, "107", nil} // beta: shuffler enabled
|
||||||
|
|
||||||
HelloIOS_Auto = HelloIOS_14
|
HelloIOS_Auto = HelloIOS_14
|
||||||
HelloIOS_11_1 = ClientHelloID{helloIOS, "111", nil} // legacy "111" means 11.1
|
HelloIOS_11_1 = ClientHelloID{helloIOS, "111", nil} // legacy "111" means 11.1
|
||||||
|
|
66
u_parrots.go
66
u_parrots.go
|
@ -10,6 +10,7 @@ import (
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
|
"math/rand"
|
||||||
"sort"
|
"sort"
|
||||||
"strconv"
|
"strconv"
|
||||||
)
|
)
|
||||||
|
@ -500,6 +501,14 @@ func utlsIdToSpec(id ClientHelloID) (ClientHelloSpec, error) {
|
||||||
&UtlsPaddingExtension{GetPaddingLen: BoringPaddingStyle},
|
&UtlsPaddingExtension{GetPaddingLen: BoringPaddingStyle},
|
||||||
},
|
},
|
||||||
}, nil
|
}, nil
|
||||||
|
case HelloChrome_107:
|
||||||
|
chs, err := utlsIdToSpec(HelloChrome_102)
|
||||||
|
if err != nil {
|
||||||
|
return chs, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Chrome 107 started shuffling the order of extensions
|
||||||
|
return shuffleExtensions(chs)
|
||||||
case HelloFirefox_55, HelloFirefox_56:
|
case HelloFirefox_55, HelloFirefox_56:
|
||||||
return ClientHelloSpec{
|
return ClientHelloSpec{
|
||||||
TLSVersMax: VersionTLS12,
|
TLSVersMax: VersionTLS12,
|
||||||
|
@ -1833,6 +1842,61 @@ func utlsIdToSpec(id ClientHelloID) (ClientHelloSpec, error) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func shuffleExtensions(chs ClientHelloSpec) (ClientHelloSpec, error) {
|
||||||
|
// Shuffle extensions to avoid fingerprinting -- introduced in Chrome 107
|
||||||
|
// GREASE, padding will remain in place (if present)
|
||||||
|
|
||||||
|
// Find indexes of GREASE and padding extensions
|
||||||
|
var greaseIdx []int
|
||||||
|
var paddingIdx []int
|
||||||
|
var otherExtensions []TLSExtension
|
||||||
|
|
||||||
|
for i, ext := range chs.Extensions {
|
||||||
|
switch ext.(type) {
|
||||||
|
case *UtlsGREASEExtension:
|
||||||
|
greaseIdx = append(greaseIdx, i)
|
||||||
|
case *UtlsPaddingExtension:
|
||||||
|
paddingIdx = append(paddingIdx, i)
|
||||||
|
default:
|
||||||
|
otherExtensions = append(otherExtensions, ext)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Shuffle other extensions
|
||||||
|
rand.Shuffle(len(otherExtensions), func(i, j int) {
|
||||||
|
otherExtensions[i], otherExtensions[j] = otherExtensions[j], otherExtensions[i]
|
||||||
|
})
|
||||||
|
|
||||||
|
// Rebuild extensions slice
|
||||||
|
otherExtIdx := 0
|
||||||
|
SHUF_EXTENSIONS:
|
||||||
|
for i := 0; i < len(chs.Extensions); i++ {
|
||||||
|
// if current index is in greaseIdx or paddingIdx, add GREASE or padding extension
|
||||||
|
for _, idx := range greaseIdx {
|
||||||
|
if i == idx {
|
||||||
|
chs.Extensions[i] = &UtlsGREASEExtension{}
|
||||||
|
continue SHUF_EXTENSIONS
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for _, idx := range paddingIdx {
|
||||||
|
if i == idx {
|
||||||
|
chs.Extensions[i] = &UtlsPaddingExtension{
|
||||||
|
GetPaddingLen: BoringPaddingStyle,
|
||||||
|
}
|
||||||
|
break SHUF_EXTENSIONS
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// otherwise add other extension
|
||||||
|
chs.Extensions[i] = otherExtensions[otherExtIdx]
|
||||||
|
otherExtIdx++
|
||||||
|
}
|
||||||
|
if otherExtIdx != len(otherExtensions) {
|
||||||
|
return ClientHelloSpec{}, errors.New("shuffleExtensions: otherExtIdx != len(otherExtensions)")
|
||||||
|
}
|
||||||
|
return chs, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (uconn *UConn) applyPresetByID(id ClientHelloID) (err error) {
|
func (uconn *UConn) applyPresetByID(id ClientHelloID) (err error) {
|
||||||
var spec ClientHelloSpec
|
var spec ClientHelloSpec
|
||||||
uconn.ClientHelloID = id
|
uconn.ClientHelloID = id
|
||||||
|
@ -1973,7 +2037,7 @@ func (uconn *UConn) ApplyPreset(p *ClientHelloSpec) error {
|
||||||
ecdheParams, err := generateECDHEParameters(uconn.config.rand(), curveID)
|
ecdheParams, err := generateECDHEParameters(uconn.config.rand(), curveID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("unsupported Curve in KeyShareExtension: %v."+
|
return fmt.Errorf("unsupported Curve in KeyShareExtension: %v."+
|
||||||
"To mimic it, fill the Data(key) field manually.", curveID)
|
"To mimic it, fill the Data(key) field manually", curveID)
|
||||||
}
|
}
|
||||||
ext.KeyShares[i].Data = ecdheParams.PublicKey()
|
ext.KeyShares[i].Data = ecdheParams.PublicKey()
|
||||||
if !preferredCurveIsSet {
|
if !preferredCurveIsSet {
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue