github.com/psiphon-labs/psiphon-tunnel-core@v2.0.28+incompatible/psiphon/common/protocol/customTLSProfiles_test.go (about)

     1  /*
     2   * Copyright (c) 2019, Psiphon Inc.
     3   * All rights reserved.
     4   *
     5   * This program is free software: you can redistribute it and/or modify
     6   * it under the terms of the GNU General Public License as published by
     7   * the Free Software Foundation, either version 3 of the License, or
     8   * (at your option) any later version.
     9   *
    10   * This program is distributed in the hope that it will be useful,
    11   * but WITHOUT ANY WARRANTY; without even the implied warranty of
    12   * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    13   * GNU General Public License for more details.
    14   *
    15   * You should have received a copy of the GNU General Public License
    16   * along with this program.  If not, see <http://www.gnu.org/licenses/>.
    17   *
    18   */
    19  
    20  package protocol
    21  
    22  import (
    23  	"bytes"
    24  	"crypto/sha256"
    25  	"encoding/json"
    26  	"testing"
    27  
    28  	utls "github.com/Psiphon-Labs/utls"
    29  )
    30  
    31  func TestCustomTLSProfiles(t *testing.T) {
    32  
    33  	// Based on utls.HelloChrome_62. Some attributes have been removed to
    34  	// eliminate randomness; and additional extensions have been added for extra
    35  	// test coverage.
    36  
    37  	utlsClientHelloSpec := &utls.ClientHelloSpec{
    38  		TLSVersMax: utls.VersionTLS12,
    39  		TLSVersMin: utls.VersionTLS10,
    40  		CipherSuites: []uint16{
    41  			utls.TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256,
    42  			utls.TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256,
    43  			utls.TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384,
    44  			utls.TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384,
    45  			utls.TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305,
    46  			utls.TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305,
    47  			utls.TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA,
    48  			utls.TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA,
    49  			utls.TLS_RSA_WITH_AES_128_GCM_SHA256,
    50  			utls.TLS_RSA_WITH_AES_256_GCM_SHA384,
    51  			utls.TLS_RSA_WITH_AES_128_CBC_SHA,
    52  			utls.TLS_RSA_WITH_AES_256_CBC_SHA,
    53  			utls.TLS_RSA_WITH_3DES_EDE_CBC_SHA,
    54  		},
    55  		CompressionMethods: []byte{0},
    56  		Extensions: []utls.TLSExtension{
    57  			&utls.RenegotiationInfoExtension{Renegotiation: utls.RenegotiateOnceAsClient},
    58  			&utls.SNIExtension{},
    59  			&utls.UtlsExtendedMasterSecretExtension{},
    60  			&utls.SessionTicketExtension{},
    61  			&utls.SignatureAlgorithmsExtension{SupportedSignatureAlgorithms: []utls.SignatureScheme{
    62  				utls.ECDSAWithP256AndSHA256,
    63  				utls.PSSWithSHA256,
    64  				utls.PKCS1WithSHA256,
    65  				utls.ECDSAWithP384AndSHA384,
    66  				utls.PSSWithSHA384,
    67  				utls.PKCS1WithSHA384,
    68  				utls.PSSWithSHA512,
    69  				utls.PKCS1WithSHA512,
    70  				utls.PKCS1WithSHA1},
    71  			},
    72  			&utls.StatusRequestExtension{},
    73  			&utls.SCTExtension{},
    74  			&utls.ALPNExtension{AlpnProtocols: []string{"h2", "http/1.1"}},
    75  			&utls.FakeChannelIDExtension{},
    76  			&utls.SupportedPointsExtension{SupportedPoints: []byte{0}},
    77  			&utls.SupportedCurvesExtension{[]utls.CurveID{
    78  				utls.X25519, utls.CurveP256, utls.CurveP384}},
    79  			&utls.UtlsPaddingExtension{GetPaddingLen: utls.BoringPaddingStyle},
    80  
    81  			// Additional extensions for test coverage
    82  			&utls.NPNExtension{NextProtos: []string{"http/1.1"}},
    83  			&utls.GenericExtension{Id: 9999, Data: []byte("generic extension")},
    84  			&utls.KeyShareExtension{[]utls.KeyShare{
    85  				{Group: utls.X25519, Data: []byte{9, 9, 9, 9}},
    86  			}},
    87  			&utls.PSKKeyExchangeModesExtension{[]uint8{
    88  				utls.PskModeDHE,
    89  			}},
    90  			&utls.SupportedVersionsExtension{[]uint16{
    91  				utls.VersionTLS13,
    92  				utls.VersionTLS12,
    93  				utls.VersionTLS11,
    94  				utls.VersionTLS10,
    95  			}},
    96  			&utls.UtlsCompressCertExtension{[]utls.CertCompressionAlgo{
    97  				utls.CertCompressionBrotli,
    98  			}},
    99  			&utls.FakeChannelIDExtension{},
   100  			&utls.FakeRecordSizeLimitExtension{Limit: 9999},
   101  		},
   102  		GetSessionID: sha256.Sum256,
   103  	}
   104  
   105  	customTLSProfilesJSON := []byte(`
   106      [
   107        {
   108          "Name": "CustomProfile",
   109          "UTLSSpec": {
   110            "TLSVersMax": 771,
   111            "TLSVersMin": 769,
   112            "CipherSuites": [49195, 49199, 49196, 49200, 52393, 52392, 49171, 49172, 156, 157, 47, 53, 10],
   113            "CompressionMethods": [0],
   114            "Extensions" : [
   115              {"Name": "RenegotiationInfo", "Data": {"Renegotiation": 1}},
   116              {"Name": "SNI"},
   117              {"Name": "ExtendedMasterSecret"},
   118              {"Name": "SessionTicket"},
   119              {"Name": "SignatureAlgorithms", "Data": {"SupportedSignatureAlgorithms": [1027, 2052, 1025, 1283, 2053, 1281, 2054, 1537, 513]}},
   120              {"Name": "StatusRequest"},
   121              {"Name": "SCT"},
   122              {"Name": "ALPN", "Data": {"AlpnProtocols": ["h2", "http/1.1"]}},
   123              {"Name": "ChannelID"},
   124              {"Name": "SupportedPoints", "Data": {"SupportedPoints": [0]}},
   125              {"Name": "SupportedCurves", "Data": {"Curves": [29, 23, 24]}},
   126              {"Name": "BoringPadding"},
   127              {"Name": "NPN", "Data": {"NextProtos": ["h2", "http/1.1"]}},
   128              {"Name": "Generic", "Data": {"Id": 9999, "Data": [103, 101, 110, 101, 114, 105, 99, 32, 101, 120, 116, 101, 110, 115, 105, 111, 110]}},
   129              {"Name": "KeyShare", "Data": {"KeyShares": [{"Group": 29, "Data": [9, 9, 9, 9]}]}},
   130              {"Name": "PSKKeyExchangeModes", "Data": {"Modes": [1]}},
   131              {"Name": "SupportedVersions", "Data": {"Versions": [772, 771, 770, 769]}},
   132              {"Name": "CertCompressionAlgs", "Data": {"Algorithms": [2]}},
   133              {"Name": "ChannelID"},
   134              {"Name": "RecordSizeLimit", "Data": {"Limit": 9999}}],
   135            "GetSessionID": "SHA-256"
   136          }
   137        }
   138      ]`)
   139  
   140  	var customTLSProfiles CustomTLSProfiles
   141  
   142  	err := json.Unmarshal(customTLSProfilesJSON, &customTLSProfiles)
   143  	if err != nil {
   144  		t.Fatalf("Unmarshal failed: %s", err)
   145  	}
   146  
   147  	err = customTLSProfiles.Validate()
   148  	if err != nil {
   149  		t.Fatalf("Validate failed: %s", err)
   150  	}
   151  
   152  	profile := customTLSProfiles[0]
   153  	profileClientHelloSpec, err := profile.GetClientHelloSpec()
   154  	if err != nil {
   155  		t.Fatalf("GetClientHelloSpec failed: %s", err)
   156  	}
   157  
   158  	zeroes := make([]byte, 32)
   159  
   160  	conn1 := utls.UClient(nil, &utls.Config{InsecureSkipVerify: true}, utls.HelloCustom)
   161  	conn1.ApplyPreset(utlsClientHelloSpec)
   162  	conn1.SetClientRandom(zeroes)
   163  	conn1.HandshakeState.Hello.SessionId = zeroes
   164  	err = conn1.BuildHandshakeState()
   165  	if err != nil {
   166  		t.Fatalf("BuildHandshakeState failed: %s", err)
   167  	}
   168  
   169  	conn2 := utls.UClient(nil, &utls.Config{InsecureSkipVerify: true}, utls.HelloCustom)
   170  	conn2.ApplyPreset(profileClientHelloSpec)
   171  	conn2.SetClientRandom(zeroes)
   172  	conn2.HandshakeState.Hello.SessionId = zeroes
   173  	err = conn2.BuildHandshakeState()
   174  	if err != nil {
   175  		t.Fatalf("BuildHandshakeState failed: %s", err)
   176  	}
   177  
   178  	if len(conn1.HandshakeState.Hello.Raw) == 0 {
   179  		t.Fatalf("Missing raw ClientHello")
   180  	}
   181  
   182  	if len(conn2.HandshakeState.Hello.Raw) == 0 {
   183  		t.Fatalf("Missing raw ClientHello")
   184  	}
   185  
   186  	if !bytes.Equal(conn1.HandshakeState.Hello.Raw, conn2.HandshakeState.Hello.Raw) {
   187  		t.Fatalf("Unidentical raw ClientHellos")
   188  	}
   189  }