udp: Add AES-based connection ID generation
This commit is contained in:
parent
c41769eb85
commit
1698f0017b
2 changed files with 94 additions and 0 deletions
50
udp/connection.go
Normal file
50
udp/connection.go
Normal file
|
@ -0,0 +1,50 @@
|
|||
// Copyright 2015 The Chihaya Authors. All rights reserved.
|
||||
// Use of this source code is governed by the BSD 2-Clause license,
|
||||
// which can be found in the LICENSE file.
|
||||
|
||||
package udp
|
||||
|
||||
import (
|
||||
"crypto/aes"
|
||||
"crypto/cipher"
|
||||
"crypto/rand"
|
||||
)
|
||||
|
||||
var connectionKey, connectionIV []byte
|
||||
|
||||
func InitConnectionIDEncryption() error {
|
||||
connectionKey = make([]byte, 16)
|
||||
_, err := rand.Read(connectionKey)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
connectionIV = make([]byte, 16)
|
||||
_, err = rand.Read(connectionIV)
|
||||
return err
|
||||
}
|
||||
|
||||
func GenerateConnectionID(ip []byte) []byte {
|
||||
block, err := aes.NewCipher(connectionKey)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
|
||||
if len(ip) > 16 {
|
||||
panic("IP larger than 16 bytes")
|
||||
}
|
||||
|
||||
for len(ip) < 8 {
|
||||
ip = append(ip, ip...) // Not enough bits in output.
|
||||
}
|
||||
|
||||
ct := make([]byte, 16)
|
||||
stream := cipher.NewCFBDecrypter(block, connectionIV)
|
||||
stream.XORKeyStream(ct, ip)
|
||||
|
||||
for i := len(ip) - 1; i >= 8; i-- {
|
||||
ct[i-8] ^= ct[i]
|
||||
}
|
||||
|
||||
return ct[:8]
|
||||
}
|
44
udp/connection_test.go
Normal file
44
udp/connection_test.go
Normal file
|
@ -0,0 +1,44 @@
|
|||
// Copyright 2015 The Chihaya Authors. All rights reserved.
|
||||
// Use of this source code is governed by the BSD 2-Clause license,
|
||||
// which can be found in the LICENSE file.
|
||||
|
||||
package udp
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"net"
|
||||
"testing"
|
||||
)
|
||||
|
||||
func TestInitReturnsNoError(t *testing.T) {
|
||||
if err := InitConnectionIDEncryption(); err != nil {
|
||||
t.Error("InitConnectionIDEncryption returned", err)
|
||||
}
|
||||
}
|
||||
|
||||
func testGenerateConnectionID(t *testing.T, ip net.IP) {
|
||||
InitConnectionIDEncryption()
|
||||
|
||||
id1 := GenerateConnectionID(ip)
|
||||
id2 := GenerateConnectionID(ip)
|
||||
|
||||
if !bytes.Equal(id1, id2) {
|
||||
t.Errorf("Connection ID mismatch: %x != %x", id1, id2)
|
||||
}
|
||||
|
||||
if len(id1) != 8 {
|
||||
t.Errorf("Connection ID had length: %d != 8", len(id1))
|
||||
}
|
||||
|
||||
if bytes.Count(id1, []byte{0}) == 8 {
|
||||
t.Errorf("Connection ID was 0")
|
||||
}
|
||||
}
|
||||
|
||||
func TestGenerateConnectionIDIPv4(t *testing.T) {
|
||||
testGenerateConnectionID(t, net.ParseIP("192.168.1.123").To4())
|
||||
}
|
||||
|
||||
func TestGenerateConnectionIDIPv6(t *testing.T) {
|
||||
testGenerateConnectionID(t, net.ParseIP("1:2:3:4::5:6"))
|
||||
}
|
Loading…
Reference in a new issue