aboutsummaryrefslogtreecommitdiffstats
path: root/p2p/peer_error_handler.go
blob: 47dcd14ff7421cc5b54cd3ae3258f781127c365e (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
package p2p

import (
    "net"
)

const (
    severityThreshold = 10
)

type DisconnectRequest struct {
    addr   net.Addr
    reason DiscReason
}

type PeerErrorHandler struct {
    quit           chan chan bool
    address        net.Addr
    peerDisconnect chan DisconnectRequest
    severity       int
    errc           chan error
}

func NewPeerErrorHandler(address net.Addr, peerDisconnect chan DisconnectRequest, errc chan error) *PeerErrorHandler {
    return &PeerErrorHandler{
        quit:           make(chan chan bool),
        address:        address,
        peerDisconnect: peerDisconnect,
        errc:           errc,
    }
}

func (self *PeerErrorHandler) Start() {
    go self.listen()
}

func (self *PeerErrorHandler) Stop() {
    q := make(chan bool)
    self.quit <- q
    <-q
}

func (self *PeerErrorHandler) listen() {
    for {
        select {
        case err, ok := <-self.errc:
            if ok {
                logger.Debugf("error %v\n", err)
                go self.handle(err)
            } else {
                return
            }
        case q := <-self.quit:
            q <- true
            return
        }
    }
}

func (self *PeerErrorHandler) handle(err error) {
    reason := DiscReason(' ')
    peerError, ok := err.(*PeerError)
    if !ok {
        peerError = NewPeerError(MiscError, " %v", err)
    }
    switch peerError.Code {
    case P2PVersionMismatch:
        reason = DiscIncompatibleVersion
    case PubkeyMissing, PubkeyInvalid:
        reason = DiscInvalidIdentity
    case PubkeyForbidden:
        reason = DiscUselessPeer
    case InvalidMsgCode, PacketTooLong, PayloadTooShort, MagicTokenMismatch, ProtocolBreach:
        reason = DiscProtocolError
    case PingTimeout:
        reason = DiscReadTimeout
    case ReadError, WriteError, MiscError:
        reason = DiscNetworkError
    case InvalidGenesis, InvalidNetworkId, InvalidProtocolVersion:
        reason = DiscSubprotocolError
    default:
        self.severity += self.getSeverity(peerError)
    }

    if self.severity >= severityThreshold {
        reason = DiscSubprotocolError
    }
    if reason != DiscReason(' ') {
        self.peerDisconnect <- DisconnectRequest{
            addr:   self.address,
            reason: reason,
        }
    }
}

func (self *PeerErrorHandler) getSeverity(peerError *PeerError) int {
    return 1
}