handshake.go 8.63 KB
Newer Older
1
// package spipe handles establishing secure communication between two peers.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
2 3 4 5 6
package spipe

import (
	"bytes"
	"errors"
7
	"fmt"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
8 9 10 11 12 13 14 15 16
	"strings"

	"crypto/aes"
	"crypto/cipher"
	"crypto/hmac"
	"crypto/rand"
	"crypto/sha1"
	"crypto/sha256"
	"crypto/sha512"
17
	bfish "github.com/jbenet/go-ipfs/Godeps/_workspace/src/code.google.com/p/go.crypto/blowfish"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
18 19 20
	"hash"

	proto "github.com/jbenet/go-ipfs/Godeps/_workspace/src/code.google.com/p/goprotobuf/proto"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
21

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
22
	ci "github.com/jbenet/go-ipfs/crypto"
23
	pb "github.com/jbenet/go-ipfs/crypto/spipe/internal/pb"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
24 25 26 27
	peer "github.com/jbenet/go-ipfs/peer"
	u "github.com/jbenet/go-ipfs/util"
)

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
28 29
var log = u.Logger("handshake")

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
30 31 32 33
// List of supported ECDH curves
var SupportedExchanges = "P-256,P-224,P-384,P-521"

// List of supported Ciphers
34
var SupportedCiphers = "AES-256,AES-128,Blowfish"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
35 36

// List of supported Hashes
37
var SupportedHashes = "SHA256,SHA512"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55

// ErrUnsupportedKeyType is returned when a private key cast/type switch fails.
var ErrUnsupportedKeyType = errors.New("unsupported key type")

// ErrClosed signals the closing of a connection.
var ErrClosed = errors.New("connection closed")

// handsahke performs initial communication over insecure channel to share
// keys, IDs, and initiate communication.
func (s *SecurePipe) handshake() error {
	// Generate and send Hello packet.
	// Hello = (rand, PublicKey, Supported)
	nonce := make([]byte, 16)
	_, err := rand.Read(nonce)
	if err != nil {
		return err
	}

Jeromy's avatar
Jeromy committed
56
	log.Debugf("handshake: %s <--> %s", s.local, s.remote)
57
	myPubKey, err := s.local.PubKey().Bytes()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
58 59 60 61
	if err != nil {
		return err
	}

62
	proposeMsg := new(pb.Propose)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
63 64 65 66 67 68 69 70 71 72 73
	proposeMsg.Rand = nonce
	proposeMsg.Pubkey = myPubKey
	proposeMsg.Exchanges = &SupportedExchanges
	proposeMsg.Ciphers = &SupportedCiphers
	proposeMsg.Hashes = &SupportedHashes

	encoded, err := proto.Marshal(proposeMsg)
	if err != nil {
		return err
	}

74 75 76 77 78 79
	// Send our Propose packet
	select {
	case s.insecure.Out <- encoded:
	case <-s.ctx.Done():
		return ErrClosed
	}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
80 81 82 83 84 85 86

	// Parse their Propose packet and generate an Exchange packet.
	// Exchange = (EphemeralPubKey, Signature)
	var resp []byte
	select {
	case <-s.ctx.Done():
		return ErrClosed
87
	case resp = <-s.insecure.In:
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
88 89
	}

90
	// u.POut("received encoded handshake\n")
91
	proposeResp := new(pb.Propose)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
92 93 94 95 96
	err = proto.Unmarshal(resp, proposeResp)
	if err != nil {
		return err
	}

97 98
	// get remote identity
	remotePubKey, err := ci.UnmarshalPublicKey(proposeResp.GetPubkey())
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
99 100 101 102
	if err != nil {
		return err
	}

103 104
	// get or construct peer
	s.remote, err = getOrConstructPeer(s.peers, remotePubKey)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
105 106 107
	if err != nil {
		return err
	}
Jeromy's avatar
Jeromy committed
108
	log.Debugf("%s Remote Peer Identified as %s", s.local, s.remote)
109

110
	exchange, err := SelectBest(SupportedExchanges, proposeResp.GetExchanges())
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
111 112 113 114
	if err != nil {
		return err
	}

115
	cipherType, err := SelectBest(SupportedCiphers, proposeResp.GetCiphers())
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
116 117 118 119
	if err != nil {
		return err
	}

120
	hashType, err := SelectBest(SupportedHashes, proposeResp.GetHashes())
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
121 122 123 124
	if err != nil {
		return err
	}

125
	// u.POut("Selected %s %s %s\n", exchange, cipherType, hashType)
126
	epubkey, genSharedKey, err := ci.GenerateEKeyPair(exchange) // Generate EphemeralPubKey
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
127 128 129 130 131 132

	var handshake bytes.Buffer // Gather corpus to sign.
	handshake.Write(encoded)
	handshake.Write(resp)
	handshake.Write(epubkey)

133
	exPacket := new(pb.Exchange)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
134 135

	exPacket.Epubkey = epubkey
136
	exPacket.Signature, err = s.local.PrivKey().Sign(handshake.Bytes())
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
137 138 139 140 141 142
	if err != nil {
		return err
	}

	exEncoded, err := proto.Marshal(exPacket)

143 144 145 146 147 148
	// send out Exchange packet
	select {
	case s.insecure.Out <- exEncoded:
	case <-s.ctx.Done():
		return ErrClosed
	}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
149 150 151 152 153 154 155 156 157 158

	// Parse their Exchange packet and generate a Finish packet.
	// Finish = E('Finish')
	var resp1 []byte
	select {
	case <-s.ctx.Done():
		return ErrClosed
	case resp1 = <-s.insecure.In:
	}

159
	exchangeResp := new(pb.Exchange)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
160 161 162 163 164 165 166 167 168 169
	err = proto.Unmarshal(resp1, exchangeResp)
	if err != nil {
		return err
	}

	var theirHandshake bytes.Buffer
	theirHandshake.Write(resp)
	theirHandshake.Write(encoded)
	theirHandshake.Write(exchangeResp.GetEpubkey())

170
	// u.POut("Remote Peer Identified as %s\n", s.remote)
171
	ok, err := s.remote.PubKey().Verify(theirHandshake.Bytes(), exchangeResp.GetSignature())
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
172 173 174 175 176 177 178 179
	if err != nil {
		return err
	}

	if !ok {
		return errors.New("Bad signature!")
	}

180
	secret, err := genSharedKey(exchangeResp.GetEpubkey())
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
181 182 183 184 185
	if err != nil {
		return err
	}

	cmp := bytes.Compare(myPubKey, proposeResp.GetPubkey())
Jeromy's avatar
Jeromy committed
186

Jeromy's avatar
Jeromy committed
187 188 189 190
	mIV, tIV, mCKey, tCKey, mMKey, tMKey := ci.KeyStretcher(cmp, cipherType, hashType, secret)

	go s.handleSecureIn(hashType, cipherType, tIV, tCKey, tMKey)
	go s.handleSecureOut(hashType, cipherType, mIV, mCKey, mMKey)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
191 192 193

	finished := []byte("Finished")

194 195 196 197 198 199 200 201
	// send finished msg
	select {
	case <-s.ctx.Done():
		return ErrClosed
	case s.Out <- finished:
	}

	// recv finished msg
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
202 203 204 205
	var resp2 []byte
	select {
	case <-s.ctx.Done():
		return ErrClosed
206
	case resp2 = <-s.In:
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
207 208 209
	}

	if bytes.Compare(resp2, finished) != 0 {
210
		return fmt.Errorf("Negotiation failed, got: %s", resp2)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
211 212
	}

Jeromy's avatar
Jeromy committed
213
	log.Debugf("%s handshake: Got node id: %s", s.local, s.remote)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
214 215 216 217 218 219 220 221 222 223 224 225 226 227
	return nil
}

func makeMac(hashType string, key []byte) (hash.Hash, int) {
	switch hashType {
	case "SHA1":
		return hmac.New(sha1.New, key), sha1.Size
	case "SHA512":
		return hmac.New(sha512.New, key), sha512.Size
	default:
		return hmac.New(sha256.New, key), sha256.Size
	}
}

228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245
func makeCipher(cipherType string, CKey []byte) (cipher.Block, error) {
	switch cipherType {
	case "AES-128", "AES-256":
		return aes.NewCipher(CKey)
	case "Blowfish":
		return bfish.NewCipher(CKey)
	default:
		return nil, fmt.Errorf("Unrecognized cipher string: %s", cipherType)
	}
}

func (s *SecurePipe) handleSecureIn(hashType, cipherType string, tIV, tCKey, tMKey []byte) {
	theirBlock, err := makeCipher(cipherType, tCKey)
	if err != nil {
		log.Criticalf("Invalid Cipher: %s", err)
		s.cancel()
		return
	}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
246 247 248 249 250
	theirCipher := cipher.NewCTR(theirBlock, tIV)

	theirMac, macSize := makeMac(hashType, tMKey)

	for {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
251 252 253 254 255 256 257 258 259
		var data []byte
		ok := true

		select {
		case <-s.ctx.Done():
			ok = false // return out
		case data, ok = <-s.insecure.In:
		}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
260
		if !ok {
261
			close(s.Duplex.In)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
262 263 264
			return
		}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
265
		// log.Debug("[peer %s] secure in [from = %s] %d", s.local, s.remote, len(data))
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
266 267 268 269 270 271 272 273 274 275 276
		if len(data) <= macSize {
			continue
		}

		mark := len(data) - macSize

		theirMac.Write(data[0:mark])
		expected := theirMac.Sum(nil)
		theirMac.Reset()

		hmacOk := hmac.Equal(data[mark:], expected)
Jeromy's avatar
Jeromy committed
277 278
		if !hmacOk {
			continue
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
279
		}
Jeromy's avatar
Jeromy committed
280 281 282 283

		theirCipher.XORKeyStream(data, data[0:mark])

		s.Duplex.In <- data[:mark]
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
284 285 286
	}
}

287 288 289 290 291 292 293
func (s *SecurePipe) handleSecureOut(hashType, cipherType string, mIV, mCKey, mMKey []byte) {
	myBlock, err := makeCipher(cipherType, mCKey)
	if err != nil {
		log.Criticalf("Invalid Cipher: %s", err)
		s.cancel()
		return
	}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
294 295 296 297 298
	myCipher := cipher.NewCTR(myBlock, mIV)

	myMac, macSize := makeMac(hashType, mMKey)

	for {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
299 300 301 302 303 304 305 306 307
		var data []byte
		ok := true

		select {
		case <-s.ctx.Done():
			ok = false // return out
		case data, ok = <-s.Out:
		}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
308
		if !ok {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
309
			close(s.insecure.Out)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
310 311 312 313 314 315 316
			return
		}

		if len(data) == 0 {
			continue
		}

Jeromy's avatar
fix bug  
Jeromy committed
317
		buff := make([]byte, len(data)+macSize)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
318

Jeromy's avatar
fix bug  
Jeromy committed
319 320 321 322 323
		myCipher.XORKeyStream(buff, data)

		myMac.Write(buff[0:len(data)])
		copy(buff[len(data):], myMac.Sum(nil))
		myMac.Reset()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
324

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
325
		// log.Debug("[peer %s] secure out [to = %s] %d", s.local, s.remote, len(buff))
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
326 327 328 329 330
		s.insecure.Out <- buff
	}
}

// Determines which algorithm to use.  Note:  f(a, b) = f(b, a)
331
func SelectBest(myPrefs, theirPrefs string) (string, error) {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
332
	// Person with greatest hash gets first choice.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
333 334
	myHash := u.Hash([]byte(myPrefs))
	theirHash := u.Hash([]byte(theirPrefs))
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359

	cmp := bytes.Compare(myHash, theirHash)
	var firstChoiceArr, secChoiceArr []string

	if cmp == -1 {
		firstChoiceArr = strings.Split(theirPrefs, ",")
		secChoiceArr = strings.Split(myPrefs, ",")
	} else if cmp == 1 {
		firstChoiceArr = strings.Split(myPrefs, ",")
		secChoiceArr = strings.Split(theirPrefs, ",")
	} else { // Exact same preferences.
		myPrefsArr := strings.Split(myPrefs, ",")
		return myPrefsArr[0], nil
	}

	for _, secChoice := range secChoiceArr {
		for _, firstChoice := range firstChoiceArr {
			if firstChoice == secChoice {
				return firstChoice, nil
			}
		}
	}

	return "", errors.New("No algorithms in common!")
}
360 361 362 363

// getOrConstructPeer attempts to fetch a peer from a peerstore.
// if succeeds, verify ID and PubKey match.
// else, construct it.
364
func getOrConstructPeer(peers peer.Peerstore, rpk ci.PubKey) (peer.Peer, error) {
365

366
	rid, err := peer.IDFromPubKey(rpk)
367 368 369 370 371
	if err != nil {
		return nil, err
	}

	npeer, err := peers.Get(rid)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
372 373
	if err != nil {
		return nil, err // unexpected error happened.
374 375
	}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
376 377 378
	// public key verification happens in Peer.VerifyAndSetPubKey
	if err := npeer.VerifyAndSetPubKey(rpk); err != nil {
		return nil, err // pubkey mismatch or other problem
379 380 381
	}
	return npeer, nil
}