2017-08-08 13:58:51 +02:00
|
|
|
package mainline
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/rand"
|
|
|
|
"crypto/sha1"
|
|
|
|
"net"
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"go.uber.org/zap"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Protocol struct {
|
|
|
|
previousTokenSecret, currentTokenSecret []byte
|
2017-11-03 00:15:13 +01:00
|
|
|
tokenLock sync.Mutex
|
|
|
|
transport *Transport
|
|
|
|
eventHandlers ProtocolEventHandlers
|
|
|
|
started bool
|
2017-08-08 13:58:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type ProtocolEventHandlers struct {
|
|
|
|
OnPingQuery func(*Message, net.Addr)
|
|
|
|
OnFindNodeQuery func(*Message, net.Addr)
|
|
|
|
OnGetPeersQuery func(*Message, net.Addr)
|
|
|
|
OnAnnouncePeerQuery func(*Message, net.Addr)
|
|
|
|
OnGetPeersResponse func(*Message, net.Addr)
|
|
|
|
OnFindNodeResponse func(*Message, net.Addr)
|
|
|
|
OnPingORAnnouncePeerResponse func(*Message, net.Addr)
|
|
|
|
}
|
|
|
|
|
2017-10-04 15:07:48 +02:00
|
|
|
func NewProtocol(laddr string, eventHandlers ProtocolEventHandlers) (p *Protocol) {
|
2017-08-08 13:58:51 +02:00
|
|
|
p = new(Protocol)
|
|
|
|
p.transport = NewTransport(laddr, p.onMessage)
|
|
|
|
p.eventHandlers = eventHandlers
|
|
|
|
|
|
|
|
p.currentTokenSecret, p.previousTokenSecret = make([]byte, 20), make([]byte, 20)
|
|
|
|
_, err := rand.Read(p.currentTokenSecret)
|
|
|
|
if err != nil {
|
|
|
|
zap.L().Fatal("Could NOT generate random bytes for token secret!", zap.Error(err))
|
|
|
|
}
|
|
|
|
copy(p.previousTokenSecret, p.currentTokenSecret)
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Protocol) Start() {
|
|
|
|
if p.started {
|
|
|
|
zap.L().Panic("Attempting to Start() a mainline/Transport that has been already started! (Programmer error.)")
|
|
|
|
}
|
|
|
|
p.started = true
|
|
|
|
|
|
|
|
p.transport.Start()
|
|
|
|
go p.updateTokenSecret()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Protocol) Terminate() {
|
|
|
|
p.transport.Terminate()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Protocol) onMessage(msg *Message, addr net.Addr) {
|
|
|
|
switch msg.Y {
|
|
|
|
case "q":
|
|
|
|
switch msg.Q {
|
|
|
|
case "ping":
|
|
|
|
if !validatePingQueryMessage(msg) {
|
|
|
|
zap.L().Debug("An invalid ping query received!")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// Check whether there is a registered event handler for the ping queries, before
|
|
|
|
// attempting to call.
|
|
|
|
if p.eventHandlers.OnPingQuery != nil {
|
|
|
|
p.eventHandlers.OnPingQuery(msg, addr)
|
|
|
|
}
|
|
|
|
|
|
|
|
case "find_node":
|
|
|
|
if !validateFindNodeQueryMessage(msg) {
|
|
|
|
zap.L().Debug("An invalid find_node query received!")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if p.eventHandlers.OnFindNodeQuery != nil {
|
|
|
|
p.eventHandlers.OnFindNodeQuery(msg, addr)
|
|
|
|
}
|
|
|
|
|
|
|
|
case "get_peers":
|
|
|
|
if !validateGetPeersQueryMessage(msg) {
|
|
|
|
zap.L().Debug("An invalid get_peers query received!")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if p.eventHandlers.OnGetPeersQuery != nil {
|
|
|
|
p.eventHandlers.OnGetPeersQuery(msg, addr)
|
|
|
|
}
|
|
|
|
|
|
|
|
case "announce_peer":
|
|
|
|
if !validateAnnouncePeerQueryMessage(msg) {
|
|
|
|
zap.L().Debug("An invalid announce_peer query received!")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if p.eventHandlers.OnAnnouncePeerQuery != nil {
|
|
|
|
p.eventHandlers.OnAnnouncePeerQuery(msg, addr)
|
|
|
|
}
|
|
|
|
|
2017-08-21 16:15:41 +02:00
|
|
|
case "vote":
|
|
|
|
// Although we are aware that such method exists, we ignore.
|
|
|
|
|
2017-08-08 13:58:51 +02:00
|
|
|
default:
|
|
|
|
zap.L().Debug("A KRPC query of an unknown method received!",
|
|
|
|
zap.String("method", msg.Q))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
case "r":
|
|
|
|
// get_peers > find_node > ping / announce_peer
|
2017-11-03 00:15:13 +01:00
|
|
|
if len(msg.R.Token) != 0 { // The message should be a get_peers response.
|
2017-08-08 13:58:51 +02:00
|
|
|
if !validateGetPeersResponseMessage(msg) {
|
|
|
|
zap.L().Debug("An invalid get_peers response received!")
|
|
|
|
return
|
|
|
|
}
|
2017-11-03 00:15:13 +01:00
|
|
|
if p.eventHandlers.OnGetPeersResponse != nil {
|
2017-08-08 13:58:51 +02:00
|
|
|
p.eventHandlers.OnGetPeersResponse(msg, addr)
|
|
|
|
}
|
2017-11-03 00:15:13 +01:00
|
|
|
} else if len(msg.R.Nodes) != 0 { // The message should be a find_node response.
|
2017-08-08 13:58:51 +02:00
|
|
|
if !validateFindNodeResponseMessage(msg) {
|
|
|
|
zap.L().Debug("An invalid find_node response received!")
|
|
|
|
return
|
|
|
|
}
|
2017-11-03 00:15:13 +01:00
|
|
|
if p.eventHandlers.OnFindNodeResponse != nil {
|
2017-08-08 13:58:51 +02:00
|
|
|
p.eventHandlers.OnFindNodeResponse(msg, addr)
|
|
|
|
}
|
2017-11-03 00:15:13 +01:00
|
|
|
} else { // The message should be a ping or an announce_peer response.
|
2017-08-08 13:58:51 +02:00
|
|
|
if !validatePingORannouncePeerResponseMessage(msg) {
|
|
|
|
zap.L().Debug("An invalid ping OR announce_peer response received!")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if p.eventHandlers.OnPingORAnnouncePeerResponse != nil {
|
|
|
|
p.eventHandlers.OnPingORAnnouncePeerResponse(msg, addr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case "e":
|
2017-11-03 00:15:13 +01:00
|
|
|
// TODO: currently ignoring Server Error 202
|
|
|
|
if msg.E.Code != 202 {
|
|
|
|
zap.L().Sugar().Debugf("Protocol error received: `%s` (%d)", msg.E.Message, msg.E.Code)
|
|
|
|
}
|
2017-08-08 13:58:51 +02:00
|
|
|
default:
|
2017-08-21 16:15:41 +02:00
|
|
|
/* zap.L().Debug("A KRPC message of an unknown type received!",
|
2017-11-03 00:15:13 +01:00
|
|
|
zap.String("type", msg.Y))
|
2017-08-21 16:15:41 +02:00
|
|
|
*/
|
2017-08-08 13:58:51 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Protocol) SendMessage(msg *Message, addr net.Addr) {
|
|
|
|
p.transport.WriteMessages(msg, addr)
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewPingQuery(id []byte) *Message {
|
|
|
|
panic("Not implemented yet!")
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewFindNodeQuery(id []byte, target []byte) *Message {
|
|
|
|
return &Message{
|
|
|
|
Y: "q",
|
2017-08-21 16:15:41 +02:00
|
|
|
T: []byte("aa"),
|
2017-08-08 13:58:51 +02:00
|
|
|
Q: "find_node",
|
|
|
|
A: QueryArguments{
|
2017-11-03 00:15:13 +01:00
|
|
|
ID: id,
|
2017-08-08 13:58:51 +02:00
|
|
|
Target: target,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewGetPeersQuery(id []byte, info_hash []byte) *Message {
|
|
|
|
panic("Not implemented yet!")
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewAnnouncePeerQuery(id []byte, implied_port bool, info_hash []byte, port uint16,
|
|
|
|
token []byte) *Message {
|
|
|
|
|
|
|
|
panic("Not implemented yet!")
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewPingResponse(t []byte, id []byte) *Message {
|
|
|
|
return &Message{
|
|
|
|
Y: "r",
|
|
|
|
T: t,
|
|
|
|
R: ResponseValues{
|
|
|
|
ID: id,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewFindNodeResponse(t []byte, id []byte, nodes []CompactNodeInfo) *Message {
|
|
|
|
panic("Not implemented yet!")
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewGetPeersResponseWithValues(t []byte, id []byte, token []byte, values []CompactPeer) *Message {
|
|
|
|
panic("Not implemented yet!")
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewGetPeersResponseWithNodes(t []byte, id []byte, token []byte, nodes []CompactNodeInfo) *Message {
|
|
|
|
return &Message{
|
|
|
|
Y: "r",
|
|
|
|
T: t,
|
|
|
|
R: ResponseValues{
|
2017-11-03 00:15:13 +01:00
|
|
|
ID: id,
|
2017-08-08 13:58:51 +02:00
|
|
|
Token: token,
|
|
|
|
Nodes: nodes,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewAnnouncePeerResponse(t []byte, id []byte) *Message {
|
|
|
|
// Because they are indistinguishable.
|
|
|
|
return NewPingResponse(t, id)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Protocol) CalculateToken(address net.IP) []byte {
|
|
|
|
p.tokenLock.Lock()
|
|
|
|
defer p.tokenLock.Unlock()
|
|
|
|
sum := sha1.Sum(append(p.currentTokenSecret, address...))
|
|
|
|
return sum[:]
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Protocol) VerifyToken(address net.IP, token []byte) bool {
|
|
|
|
p.tokenLock.Lock()
|
|
|
|
defer p.tokenLock.Unlock()
|
|
|
|
panic("VerifyToken() not implemented yet!")
|
|
|
|
// TODO
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Protocol) updateTokenSecret() {
|
|
|
|
for range time.Tick(10 * time.Minute) {
|
|
|
|
p.tokenLock.Lock()
|
|
|
|
copy(p.previousTokenSecret, p.currentTokenSecret)
|
|
|
|
_, err := rand.Read(p.currentTokenSecret)
|
|
|
|
if err != nil {
|
|
|
|
p.tokenLock.Unlock()
|
|
|
|
zap.L().Fatal("Could NOT generate random bytes for token secret!", zap.Error(err))
|
|
|
|
}
|
|
|
|
p.tokenLock.Unlock()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func validatePingQueryMessage(msg *Message) bool {
|
|
|
|
return len(msg.A.ID) == 20
|
|
|
|
}
|
|
|
|
|
|
|
|
func validateFindNodeQueryMessage(msg *Message) bool {
|
|
|
|
return len(msg.A.ID) == 20 &&
|
|
|
|
len(msg.A.Target) == 20
|
|
|
|
}
|
|
|
|
|
|
|
|
func validateGetPeersQueryMessage(msg *Message) bool {
|
|
|
|
return len(msg.A.ID) == 20 &&
|
|
|
|
len(msg.A.InfoHash) == 20
|
|
|
|
}
|
|
|
|
|
|
|
|
func validateAnnouncePeerQueryMessage(msg *Message) bool {
|
|
|
|
return len(msg.A.ID) == 20 &&
|
|
|
|
len(msg.A.InfoHash) == 20 &&
|
|
|
|
msg.A.Port > 0 &&
|
|
|
|
len(msg.A.Token) > 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func validatePingORannouncePeerResponseMessage(msg *Message) bool {
|
|
|
|
return len(msg.R.ID) == 20
|
|
|
|
}
|
|
|
|
|
|
|
|
func validateFindNodeResponseMessage(msg *Message) bool {
|
|
|
|
if len(msg.R.ID) != 20 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: check nodes field
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func validateGetPeersResponseMessage(msg *Message) bool {
|
|
|
|
return len(msg.R.ID) == 20 &&
|
|
|
|
len(msg.R.Token) > 0
|
|
|
|
|
|
|
|
// TODO: check for values or nodes
|
|
|
|
}
|