| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // Copyright 2015 The go-ethereum Authors | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // This file is part of the go-ethereum library. | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // | 
					
						
							| 
									
										
										
										
											2015-07-23 18:35:11 +02:00
										 |  |  | // The go-ethereum library is free software: you can redistribute it and/or modify | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // it under the terms of the GNU Lesser General Public License as published by | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | // (at your option) any later version. | 
					
						
							|  |  |  | // | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // The go-ethereum library is distributed in the hope that it will be useful, | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // GNU Lesser General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-21 20:23:37 +03:00
										 |  |  | // Contains the meters and timers used by the networking layer. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package p2p | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2015-06-21 20:23:37 +03:00
										 |  |  | 	"net" | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 	"sync" | 
					
						
							|  |  |  | 	"sync/atomic" | 
					
						
							|  |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2015-06-21 20:23:37 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 	"github.com/ethereum/go-ethereum/event" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/log" | 
					
						
							| 
									
										
										
										
											2015-06-29 16:11:01 +03:00
										 |  |  | 	"github.com/ethereum/go-ethereum/metrics" | 
					
						
							| 
									
										
										
										
											2019-06-10 14:21:02 +03:00
										 |  |  | 	"github.com/ethereum/go-ethereum/p2p/enode" | 
					
						
							| 
									
										
										
										
											2015-06-21 20:23:37 +03:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | const ( | 
					
						
							| 
									
										
										
										
											2019-06-10 14:21:02 +03:00
										 |  |  | 	MetricsInboundTraffic   = "p2p/ingress" // Name for the registered inbound traffic meter | 
					
						
							|  |  |  | 	MetricsOutboundTraffic  = "p2p/egress"  // Name for the registered outbound traffic meter | 
					
						
							|  |  |  | 	MetricsOutboundConnects = "p2p/dials"   // Name for the registered outbound connects meter | 
					
						
							|  |  |  | 	MetricsInboundConnects  = "p2p/serves"  // Name for the registered inbound connects meter | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	MeteredPeerLimit = 1024 // This amount of peers are individually metered | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-21 20:23:37 +03:00
										 |  |  | var ( | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 	ingressConnectMeter = metrics.NewRegisteredMeter(MetricsInboundConnects, nil)  // Meter counting the ingress connections | 
					
						
							|  |  |  | 	ingressTrafficMeter = metrics.NewRegisteredMeter(MetricsInboundTraffic, nil)   // Meter metering the cumulative ingress traffic | 
					
						
							|  |  |  | 	egressConnectMeter  = metrics.NewRegisteredMeter(MetricsOutboundConnects, nil) // Meter counting the egress connections | 
					
						
							|  |  |  | 	egressTrafficMeter  = metrics.NewRegisteredMeter(MetricsOutboundTraffic, nil)  // Meter metering the cumulative egress traffic | 
					
						
							| 
									
										
										
										
											2019-09-10 14:39:07 +03:00
										 |  |  | 	activePeerGauge     = metrics.NewRegisteredGauge("p2p/peers", nil)             // Gauge tracking the current peer count | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-09 11:20:51 +02:00
										 |  |  | 	PeerIngressRegistry = metrics.NewPrefixedChildRegistry(metrics.EphemeralRegistry, MetricsInboundTraffic+"/")  // Registry containing the peer ingress | 
					
						
							|  |  |  | 	PeerEgressRegistry  = metrics.NewPrefixedChildRegistry(metrics.EphemeralRegistry, MetricsOutboundTraffic+"/") // Registry containing the peer egress | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	meteredPeerFeed  event.Feed // Event feed for peer metrics | 
					
						
							|  |  |  | 	meteredPeerCount int32      // Actually stored peer connection count | 
					
						
							| 
									
										
										
										
											2015-06-21 20:23:37 +03:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | // MeteredPeerEventType is the type of peer events emitted by a metered connection. | 
					
						
							|  |  |  | type MeteredPeerEventType int | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const ( | 
					
						
							|  |  |  | 	// PeerConnected is the type of event emitted when a peer successfully | 
					
						
							|  |  |  | 	// made the handshake. | 
					
						
							|  |  |  | 	PeerConnected MeteredPeerEventType = iota | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// PeerDisconnected is the type of event emitted when a peer disconnects. | 
					
						
							|  |  |  | 	PeerDisconnected | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// PeerHandshakeFailed is the type of event emitted when a peer fails to | 
					
						
							|  |  |  | 	// make the handshake or disconnects before the handshake. | 
					
						
							|  |  |  | 	PeerHandshakeFailed | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // MeteredPeerEvent is an event emitted when peers connect or disconnect. | 
					
						
							|  |  |  | type MeteredPeerEvent struct { | 
					
						
							|  |  |  | 	Type    MeteredPeerEventType // Type of peer event | 
					
						
							|  |  |  | 	IP      net.IP               // IP address of the peer | 
					
						
							| 
									
										
										
										
											2018-11-08 13:11:20 +02:00
										 |  |  | 	ID      enode.ID             // NodeID of the peer | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 	Elapsed time.Duration        // Time elapsed between the connection and the handshake/disconnection | 
					
						
							|  |  |  | 	Ingress uint64               // Ingress count at the moment of the event | 
					
						
							|  |  |  | 	Egress  uint64               // Egress count at the moment of the event | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // SubscribeMeteredPeerEvent registers a subscription for peer life-cycle events | 
					
						
							|  |  |  | // if metrics collection is enabled. | 
					
						
							|  |  |  | func SubscribeMeteredPeerEvent(ch chan<- MeteredPeerEvent) event.Subscription { | 
					
						
							|  |  |  | 	return meteredPeerFeed.Subscribe(ch) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | // meteredConn is a wrapper around a net.Conn that meters both the | 
					
						
							| 
									
										
										
										
											2015-06-21 20:23:37 +03:00
										 |  |  | // inbound and outbound network traffic. | 
					
						
							|  |  |  | type meteredConn struct { | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 	net.Conn // Network connection to wrap with metering | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	connected time.Time // Connection time of the peer | 
					
						
							|  |  |  | 	ip        net.IP    // IP address of the peer | 
					
						
							| 
									
										
										
										
											2018-11-08 13:11:20 +02:00
										 |  |  | 	id        enode.ID  // NodeID of the peer | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// trafficMetered denotes if the peer is registered in the traffic registries. | 
					
						
							|  |  |  | 	// Its value is true if the metered peer count doesn't reach the limit in the | 
					
						
							|  |  |  | 	// moment of the peer's connection. | 
					
						
							|  |  |  | 	trafficMetered bool | 
					
						
							|  |  |  | 	ingressMeter   metrics.Meter // Meter for the read bytes of the peer | 
					
						
							|  |  |  | 	egressMeter    metrics.Meter // Meter for the written bytes of the peer | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	lock sync.RWMutex // Lock protecting the metered connection's internals | 
					
						
							| 
									
										
										
										
											2015-06-21 20:23:37 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | // newMeteredConn creates a new metered connection, bumps the ingress or egress | 
					
						
							|  |  |  | // connection meter and also increases the metered peer count. If the metrics | 
					
						
							|  |  |  | // system is disabled or the IP address is unspecified, this function returns | 
					
						
							|  |  |  | // the original object. | 
					
						
							|  |  |  | func newMeteredConn(conn net.Conn, ingress bool, ip net.IP) net.Conn { | 
					
						
							| 
									
										
										
										
											2015-07-02 14:13:46 +03:00
										 |  |  | 	// Short circuit if metrics are disabled | 
					
						
							|  |  |  | 	if !metrics.Enabled { | 
					
						
							|  |  |  | 		return conn | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 	if ip.IsUnspecified() { | 
					
						
							|  |  |  | 		log.Warn("Peer IP is unspecified") | 
					
						
							|  |  |  | 		return conn | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Bump the connection counters and wrap the connection | 
					
						
							| 
									
										
										
										
											2015-06-21 20:23:37 +03:00
										 |  |  | 	if ingress { | 
					
						
							|  |  |  | 		ingressConnectMeter.Mark(1) | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		egressConnectMeter.Mark(1) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-09-10 14:39:07 +03:00
										 |  |  | 	activePeerGauge.Inc(1) | 
					
						
							| 
									
										
										
										
											2019-06-10 14:21:02 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 	return &meteredConn{ | 
					
						
							|  |  |  | 		Conn:      conn, | 
					
						
							|  |  |  | 		ip:        ip, | 
					
						
							|  |  |  | 		connected: time.Now(), | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-06-21 20:23:37 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | // Read delegates a network read to the underlying connection, bumping the common | 
					
						
							|  |  |  | // and the peer ingress traffic meters along the way. | 
					
						
							| 
									
										
										
										
											2015-06-21 20:23:37 +03:00
										 |  |  | func (c *meteredConn) Read(b []byte) (n int, err error) { | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 	n, err = c.Conn.Read(b) | 
					
						
							| 
									
										
										
										
											2015-06-21 20:23:37 +03:00
										 |  |  | 	ingressTrafficMeter.Mark(int64(n)) | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 	c.lock.RLock() | 
					
						
							|  |  |  | 	if c.trafficMetered { | 
					
						
							|  |  |  | 		c.ingressMeter.Mark(int64(n)) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	c.lock.RUnlock() | 
					
						
							|  |  |  | 	return n, err | 
					
						
							| 
									
										
										
										
											2015-06-21 20:23:37 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | // Write delegates a network write to the underlying connection, bumping the common | 
					
						
							|  |  |  | // and the peer egress traffic meters along the way. | 
					
						
							| 
									
										
										
										
											2015-06-21 20:23:37 +03:00
										 |  |  | func (c *meteredConn) Write(b []byte) (n int, err error) { | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 	n, err = c.Conn.Write(b) | 
					
						
							| 
									
										
										
										
											2015-06-21 20:23:37 +03:00
										 |  |  | 	egressTrafficMeter.Mark(int64(n)) | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 	c.lock.RLock() | 
					
						
							|  |  |  | 	if c.trafficMetered { | 
					
						
							|  |  |  | 		c.egressMeter.Mark(int64(n)) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	c.lock.RUnlock() | 
					
						
							|  |  |  | 	return n, err | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // handshakeDone is called when a peer handshake is done. Registers the peer to | 
					
						
							|  |  |  | // the ingress and the egress traffic registries using the peer's IP and node ID, | 
					
						
							|  |  |  | // also emits connect event. | 
					
						
							| 
									
										
										
										
											2018-11-08 13:11:20 +02:00
										 |  |  | func (c *meteredConn) handshakeDone(id enode.ID) { | 
					
						
							| 
									
										
										
										
											2019-03-13 14:53:52 +02:00
										 |  |  | 	// TODO (kurkomisi): use the node URL instead of the pure node ID. (the String() method of *Node) | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 	if atomic.AddInt32(&meteredPeerCount, 1) >= MeteredPeerLimit { | 
					
						
							|  |  |  | 		// Don't register the peer in the traffic registries. | 
					
						
							|  |  |  | 		atomic.AddInt32(&meteredPeerCount, -1) | 
					
						
							|  |  |  | 		c.lock.Lock() | 
					
						
							|  |  |  | 		c.id, c.trafficMetered = id, false | 
					
						
							|  |  |  | 		c.lock.Unlock() | 
					
						
							|  |  |  | 		log.Warn("Metered peer count reached the limit") | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2018-11-08 13:11:20 +02:00
										 |  |  | 		key := fmt.Sprintf("%s/%s", c.ip, id.String()) | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 		c.lock.Lock() | 
					
						
							|  |  |  | 		c.id, c.trafficMetered = id, true | 
					
						
							|  |  |  | 		c.ingressMeter = metrics.NewRegisteredMeter(key, PeerIngressRegistry) | 
					
						
							|  |  |  | 		c.egressMeter = metrics.NewRegisteredMeter(key, PeerEgressRegistry) | 
					
						
							|  |  |  | 		c.lock.Unlock() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	meteredPeerFeed.Send(MeteredPeerEvent{ | 
					
						
							|  |  |  | 		Type:    PeerConnected, | 
					
						
							|  |  |  | 		IP:      c.ip, | 
					
						
							|  |  |  | 		ID:      id, | 
					
						
							|  |  |  | 		Elapsed: time.Since(c.connected), | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Close delegates a close operation to the underlying connection, unregisters | 
					
						
							|  |  |  | // the peer from the traffic registries and emits close event. | 
					
						
							|  |  |  | func (c *meteredConn) Close() error { | 
					
						
							|  |  |  | 	err := c.Conn.Close() | 
					
						
							|  |  |  | 	c.lock.RLock() | 
					
						
							| 
									
										
										
										
											2018-11-08 13:11:20 +02:00
										 |  |  | 	if c.id == (enode.ID{}) { | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 		// If the peer disconnects before the handshake. | 
					
						
							|  |  |  | 		c.lock.RUnlock() | 
					
						
							|  |  |  | 		meteredPeerFeed.Send(MeteredPeerEvent{ | 
					
						
							|  |  |  | 			Type:    PeerHandshakeFailed, | 
					
						
							|  |  |  | 			IP:      c.ip, | 
					
						
							|  |  |  | 			Elapsed: time.Since(c.connected), | 
					
						
							|  |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2019-09-10 14:39:07 +03:00
										 |  |  | 		activePeerGauge.Dec(1) | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	id := c.id | 
					
						
							|  |  |  | 	if !c.trafficMetered { | 
					
						
							|  |  |  | 		// If the peer isn't registered in the traffic registries. | 
					
						
							|  |  |  | 		c.lock.RUnlock() | 
					
						
							|  |  |  | 		meteredPeerFeed.Send(MeteredPeerEvent{ | 
					
						
							|  |  |  | 			Type: PeerDisconnected, | 
					
						
							|  |  |  | 			IP:   c.ip, | 
					
						
							|  |  |  | 			ID:   id, | 
					
						
							|  |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2019-09-10 14:39:07 +03:00
										 |  |  | 		activePeerGauge.Dec(1) | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ingress, egress := uint64(c.ingressMeter.Count()), uint64(c.egressMeter.Count()) | 
					
						
							|  |  |  | 	c.lock.RUnlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Decrement the metered peer count | 
					
						
							|  |  |  | 	atomic.AddInt32(&meteredPeerCount, -1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Unregister the peer from the traffic registries | 
					
						
							|  |  |  | 	key := fmt.Sprintf("%s/%s", c.ip, id) | 
					
						
							|  |  |  | 	PeerIngressRegistry.Unregister(key) | 
					
						
							|  |  |  | 	PeerEgressRegistry.Unregister(key) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	meteredPeerFeed.Send(MeteredPeerEvent{ | 
					
						
							|  |  |  | 		Type:    PeerDisconnected, | 
					
						
							|  |  |  | 		IP:      c.ip, | 
					
						
							|  |  |  | 		ID:      id, | 
					
						
							|  |  |  | 		Ingress: ingress, | 
					
						
							|  |  |  | 		Egress:  egress, | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2019-09-10 14:39:07 +03:00
										 |  |  | 	activePeerGauge.Dec(1) | 
					
						
							| 
									
										
										
										
											2018-10-16 01:40:51 +03:00
										 |  |  | 	return err | 
					
						
							| 
									
										
										
										
											2015-06-21 20:23:37 +03:00
										 |  |  | } |