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

     1  /*
     2   * Copyright (c) 2015, 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  	"encoding/hex"
    25  	"encoding/json"
    26  	"strconv"
    27  	"testing"
    28  
    29  	"github.com/Psiphon-Labs/psiphon-tunnel-core/psiphon/common"
    30  	"github.com/Psiphon-Labs/psiphon-tunnel-core/psiphon/common/prng"
    31  )
    32  
    33  const (
    34  	_VALID_NORMAL_SERVER_ENTRY                    = `192.168.0.1 80 <webServerSecret> <webServerCertificate> {"ipAddress":"192.168.0.1","webServerPort":"80","webServerSecret":"<webServerSecret>","webServerCertificate":"<webServerCertificate>","sshPort":22,"sshUsername":"<sshUsername>","sshPassword":"<sshPassword>","sshHostKey":"<sshHostKey>","sshObfuscatedPort":443,"sshObfuscatedKey":"<sshObfuscatedKey>","capabilities":["handshake","SSH","OSSH","VPN"],"region":"CA","meekServerPort":8080,"meekCookieEncryptionPublicKey":"<meekCookieEncryptionPublicKey>","meekObfuscatedKey":"<meekObfuscatedKey>","meekFrontingDomain":"<meekFrontingDomain>","meekFrontingHost":"<meekFrontingHost>"}`
    35  	_VALID_BLANK_LEGACY_SERVER_ENTRY              = `    {"ipAddress":"192.168.0.1","webServerPort":"80","webServerSecret":"<webServerSecret>","webServerCertificate":"<webServerCertificate>","sshPort":22,"sshUsername":"<sshUsername>","sshPassword":"<sshPassword>","sshHostKey":"<sshHostKey>","sshObfuscatedPort":443,"sshObfuscatedKey":"<sshObfuscatedKey>","capabilities":["handshake","SSH","OSSH","VPN"],"region":"CA","meekServerPort":8080,"meekCookieEncryptionPublicKey":"<meekCookieEncryptionPublicKey>","meekObfuscatedKey":"<meekObfuscatedKey>","meekFrontingDomain":"<meekFrontingDomain>","meekFrontingHost":"<meekFrontingHost>"}`
    36  	_VALID_FUTURE_SERVER_ENTRY                    = `192.168.0.1 80 <webServerSecret> <webServerCertificate> {"ipAddress":"192.168.0.1","webServerPort":"80","webServerSecret":"<webServerSecret>","webServerCertificate":"<webServerCertificate>","sshPort":22,"sshUsername":"<sshUsername>","sshPassword":"<sshPassword>","sshHostKey":"<sshHostKey>","sshObfuscatedPort":443,"sshObfuscatedKey":"<sshObfuscatedKey>","capabilities":["handshake","SSH","OSSH","VPN"],"region":"CA","meekServerPort":8080,"meekCookieEncryptionPublicKey":"<meekCookieEncryptionPublicKey>","meekObfuscatedKey":"<meekObfuscatedKey>","meekFrontingDomain":"<meekFrontingDomain>","meekFrontingHost":"<meekFrontingHost>","dummyFutureField":"dummyFutureField"}`
    37  	_INVALID_WINDOWS_REGISTRY_LEGACY_SERVER_ENTRY = `192.168.0.1 80 <webServerSecret> <webServerCertificate> {"sshPort":22,"sshUsername":"<sshUsername>","sshPassword":"<sshPassword>","sshHostKey":"<sshHostKey>","sshObfuscatedPort":443,"sshObfuscatedKey":"<sshObfuscatedKey>","capabilities":["handshake","SSH","OSSH","VPN"],"region":"CA","meekServerPort":8080,"meekCookieEncryptionPublicKey":"<meekCookieEncryptionPublicKey>","meekObfuscatedKey":"<meekObfuscatedKey>","meekFrontingDomain":"<meekFrontingDomain>","meekFrontingHost":"<meekFrontingHost>"}`
    38  	_INVALID_MALFORMED_IP_ADDRESS_SERVER_ENTRY    = `192.168.0.1 80 <webServerSecret> <webServerCertificate> {"ipAddress":"192.168.0.","webServerPort":"80","webServerSecret":"<webServerSecret>","webServerCertificate":"<webServerCertificate>","sshPort":22,"sshUsername":"<sshUsername>","sshPassword":"<sshPassword>","sshHostKey":"<sshHostKey>","sshObfuscatedPort":443,"sshObfuscatedKey":"<sshObfuscatedKey>","capabilities":["handshake","SSH","OSSH","VPN"],"region":"CA","meekServerPort":8080,"meekCookieEncryptionPublicKey":"<meekCookieEncryptionPublicKey>","meekObfuscatedKey":"<meekObfuscatedKey>","meekFrontingDomain":"<meekFrontingDomain>","meekFrontingHost":"<meekFrontingHost>"}`
    39  	_EXPECTED_IP_ADDRESS                          = `192.168.0.1`
    40  	_EXPECTED_DUMMY_FUTURE_FIELD                  = `dummyFutureField`
    41  )
    42  
    43  var testEncodedServerEntryList = hex.EncodeToString([]byte(_VALID_NORMAL_SERVER_ENTRY)) + "\n" +
    44  	hex.EncodeToString([]byte(_VALID_BLANK_LEGACY_SERVER_ENTRY)) + "\n" +
    45  	hex.EncodeToString([]byte(_VALID_FUTURE_SERVER_ENTRY)) + "\n" +
    46  	hex.EncodeToString([]byte(_INVALID_WINDOWS_REGISTRY_LEGACY_SERVER_ENTRY)) + "\n" +
    47  	hex.EncodeToString([]byte(_INVALID_MALFORMED_IP_ADDRESS_SERVER_ENTRY))
    48  
    49  // DecodeServerEntryList should return 3 valid decoded entries from the input list of 5
    50  func TestDecodeServerEntryList(t *testing.T) {
    51  
    52  	serverEntries, err := DecodeServerEntryList(
    53  		testEncodedServerEntryList, common.GetCurrentTimestamp(), SERVER_ENTRY_SOURCE_EMBEDDED)
    54  	if err != nil {
    55  		t.Error(err.Error())
    56  		t.FailNow()
    57  	}
    58  
    59  	if len(serverEntries) != 3 {
    60  		t.Error("unexpected number of valid server entries")
    61  	}
    62  
    63  	numFutureFields := 0
    64  
    65  	for _, serverEntryFields := range serverEntries {
    66  		if serverEntryFields.GetIPAddress() != _EXPECTED_IP_ADDRESS {
    67  			t.Errorf("unexpected IP address in decoded server entry: %s", serverEntryFields.GetIPAddress())
    68  		}
    69  		if futureField, ok := serverEntryFields[_EXPECTED_DUMMY_FUTURE_FIELD]; ok {
    70  			if futureFieldStr, ok := futureField.(string); ok && futureFieldStr == _EXPECTED_DUMMY_FUTURE_FIELD {
    71  				numFutureFields += 1
    72  			}
    73  		}
    74  	}
    75  
    76  	if numFutureFields != 1 {
    77  		t.Error("unexpected number of retained future fields")
    78  	}
    79  }
    80  
    81  func TestStreamingServerEntryDecoder(t *testing.T) {
    82  
    83  	decoder := NewStreamingServerEntryDecoder(
    84  		bytes.NewReader([]byte(testEncodedServerEntryList)),
    85  		common.GetCurrentTimestamp(), SERVER_ENTRY_SOURCE_EMBEDDED)
    86  
    87  	serverEntries := make([]ServerEntryFields, 0)
    88  
    89  	for {
    90  		serverEntryFields, err := decoder.Next()
    91  		if err != nil {
    92  			t.Error(err.Error())
    93  			t.FailNow()
    94  		}
    95  
    96  		if serverEntryFields == nil {
    97  			break
    98  		}
    99  
   100  		serverEntries = append(serverEntries, serverEntryFields)
   101  	}
   102  
   103  	if len(serverEntries) != 3 {
   104  		t.Error("unexpected number of valid server entries")
   105  	}
   106  
   107  	numFutureFields := 0
   108  
   109  	for _, serverEntryFields := range serverEntries {
   110  		if serverEntryFields.GetIPAddress() != _EXPECTED_IP_ADDRESS {
   111  			t.Errorf("unexpected IP address in decoded server entry: %s", serverEntryFields.GetIPAddress())
   112  		}
   113  		if futureField, ok := serverEntryFields[_EXPECTED_DUMMY_FUTURE_FIELD]; ok {
   114  			if futureFieldStr, ok := futureField.(string); ok && futureFieldStr == _EXPECTED_DUMMY_FUTURE_FIELD {
   115  				numFutureFields += 1
   116  			}
   117  		}
   118  	}
   119  
   120  	if numFutureFields != 1 {
   121  		t.Error("unexpected number of retained future fields")
   122  	}
   123  }
   124  
   125  // Directly call DecodeServerEntryFields and ValidateServerEntry with invalid inputs
   126  func TestInvalidServerEntries(t *testing.T) {
   127  
   128  	testCases := [2]string{_INVALID_WINDOWS_REGISTRY_LEGACY_SERVER_ENTRY, _INVALID_MALFORMED_IP_ADDRESS_SERVER_ENTRY}
   129  
   130  	for _, testCase := range testCases {
   131  		encodedServerEntry := hex.EncodeToString([]byte(testCase))
   132  		serverEntryFields, err := DecodeServerEntryFields(
   133  			encodedServerEntry, common.GetCurrentTimestamp(), SERVER_ENTRY_SOURCE_EMBEDDED)
   134  		if err != nil {
   135  			t.Error(err.Error())
   136  		}
   137  		err = ValidateServerEntryFields(serverEntryFields)
   138  		if err == nil {
   139  			t.Errorf("server entry should not validate: %s", testCase)
   140  		}
   141  	}
   142  }
   143  
   144  // Directly call DecodeServerEntry
   145  func TestDecodeServerEntryStruct(t *testing.T) {
   146  
   147  	serverEntry, err := DecodeServerEntry(
   148  		hex.EncodeToString([]byte(_VALID_NORMAL_SERVER_ENTRY)),
   149  		common.GetCurrentTimestamp(), SERVER_ENTRY_SOURCE_EMBEDDED)
   150  	if err != nil {
   151  		t.Error(err.Error())
   152  		t.FailNow()
   153  	}
   154  	if serverEntry.IpAddress != _EXPECTED_IP_ADDRESS {
   155  		t.Errorf("unexpected IP address in decoded server entry: %s", serverEntry.IpAddress)
   156  	}
   157  }
   158  
   159  func TestServerEntryListSignatures(t *testing.T) {
   160  	testServerEntryListSignatures(t, true)
   161  	testServerEntryListSignatures(t, false)
   162  }
   163  
   164  func testServerEntryListSignatures(t *testing.T, setExplicitTag bool) {
   165  
   166  	publicKey, privateKey, err := NewServerEntrySignatureKeyPair()
   167  	if err != nil {
   168  		t.Fatalf("NewServerEntrySignatureKeyPair failed: %s", err)
   169  	}
   170  
   171  	n := 16
   172  	serverEntry := &ServerEntry{
   173  		IpAddress:                     prng.HexString(n),
   174  		WebServerPort:                 strconv.Itoa(prng.Intn(n)),
   175  		WebServerSecret:               prng.HexString(n),
   176  		WebServerCertificate:          prng.HexString(n),
   177  		SshPort:                       prng.Intn(n),
   178  		SshUsername:                   prng.HexString(n),
   179  		SshPassword:                   prng.HexString(n),
   180  		SshHostKey:                    prng.HexString(n),
   181  		SshObfuscatedPort:             prng.Intn(n),
   182  		SshObfuscatedQUICPort:         prng.Intn(n),
   183  		SshObfuscatedTapDancePort:     prng.Intn(n),
   184  		SshObfuscatedConjurePort:      prng.Intn(n),
   185  		SshObfuscatedKey:              prng.HexString(n),
   186  		Capabilities:                  []string{prng.HexString(n)},
   187  		Region:                        prng.HexString(n),
   188  		MeekServerPort:                prng.Intn(n),
   189  		MeekCookieEncryptionPublicKey: prng.HexString(n),
   190  		MeekObfuscatedKey:             prng.HexString(n),
   191  		MeekFrontingHost:              prng.HexString(n),
   192  		MeekFrontingHosts:             []string{prng.HexString(n)},
   193  		MeekFrontingDomain:            prng.HexString(n),
   194  		MeekFrontingAddresses:         []string{prng.HexString(n)},
   195  		MeekFrontingAddressesRegex:    prng.HexString(n),
   196  		MeekFrontingDisableSNI:        false,
   197  		TacticsRequestPublicKey:       prng.HexString(n),
   198  		TacticsRequestObfuscatedKey:   prng.HexString(n),
   199  		ConfigurationVersion:          1,
   200  	}
   201  
   202  	if setExplicitTag {
   203  		serverEntry.Tag = prng.HexString(n)
   204  	}
   205  
   206  	// Convert ServerEntry to ServerEntryFields
   207  
   208  	marshaledServerEntry, err := json.Marshal(serverEntry)
   209  	if err != nil {
   210  		t.Fatalf("Marshal failed: %s", err)
   211  	}
   212  
   213  	var serverEntryFields ServerEntryFields
   214  
   215  	err = json.Unmarshal(marshaledServerEntry, &serverEntryFields)
   216  	if err != nil {
   217  		t.Fatalf("Unmarshal failed: %s", err)
   218  	}
   219  
   220  	// Check that local fields are ignored in the signature
   221  
   222  	if !setExplicitTag {
   223  		serverEntryFields.SetTag(prng.HexString(n))
   224  	}
   225  	serverEntryFields.SetLocalSource(prng.HexString(n))
   226  	serverEntryFields.SetLocalTimestamp(prng.HexString(n))
   227  
   228  	// Set dummy signature to check that its overwritten
   229  
   230  	serverEntryFields["signature"] = prng.HexString(n)
   231  
   232  	err = serverEntryFields.AddSignature(publicKey, privateKey)
   233  	if err != nil {
   234  		t.Fatalf("AddSignature failed: %s", err)
   235  	}
   236  
   237  	err = serverEntryFields.VerifySignature(publicKey)
   238  	if err != nil {
   239  		t.Fatalf("VerifySignature failed: %s", err)
   240  	}
   241  
   242  	// A 2nd VerifySignature call checks that the first VerifySignature
   243  	// call leaves the server entry fields intact
   244  
   245  	err = serverEntryFields.VerifySignature(publicKey)
   246  	if err != nil {
   247  		t.Fatalf("VerifySignature failed: %s", err)
   248  	}
   249  
   250  	// Modify local local fields and check that signature remains valid
   251  
   252  	if !setExplicitTag {
   253  		serverEntryFields.SetTag(prng.HexString(n))
   254  	}
   255  	serverEntryFields.SetLocalSource(prng.HexString(n))
   256  	serverEntryFields.SetLocalTimestamp(prng.HexString(n))
   257  
   258  	err = serverEntryFields.VerifySignature(publicKey)
   259  	if err != nil {
   260  		t.Fatalf("VerifySignature failed: %s", err)
   261  	}
   262  
   263  	// Check that verification fails when using the wrong public key
   264  
   265  	incorrectPublicKey, _, err := NewServerEntrySignatureKeyPair()
   266  	if err != nil {
   267  		t.Fatalf("NewServerEntrySignatureKeyPair failed: %s", err)
   268  	}
   269  
   270  	err = serverEntryFields.VerifySignature(incorrectPublicKey)
   271  	if err == nil {
   272  		t.Fatalf("VerifySignature unexpectedly succeeded")
   273  	}
   274  
   275  	// Check that an expected, non-local field causes verification to fail
   276  
   277  	serverEntryFields[prng.HexString(n)] = prng.HexString(n)
   278  
   279  	err = serverEntryFields.VerifySignature(publicKey)
   280  	if err == nil {
   281  		t.Fatalf("AddSignature unexpectedly succeeded")
   282  	}
   283  
   284  	// Check that modifying a signed field causes verification to fail
   285  
   286  	fieldName := "sshObfuscatedKey"
   287  	if setExplicitTag {
   288  		fieldName = "tag"
   289  	}
   290  
   291  	serverEntryFields[fieldName] = prng.HexString(n)
   292  
   293  	err = serverEntryFields.VerifySignature(publicKey)
   294  	if err == nil {
   295  		t.Fatalf("AddSignature unexpectedly succeeded")
   296  	}
   297  }