mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-19 00:04:12 +00:00
188 lines
5.4 KiB
Go
188 lines
5.4 KiB
Go
package node
|
|
|
|
import (
|
|
"encoding/json"
|
|
"io"
|
|
"net/http"
|
|
"strings"
|
|
|
|
corenet "github.com/libp2p/go-libp2p/core/network"
|
|
"github.com/libp2p/go-libp2p/core/peer"
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/p2p"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/p2p/peers"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/p2p/peers/peerdata"
|
|
"github.com/prysmaticlabs/prysm/v4/network/httputil"
|
|
eth "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1"
|
|
"go.opencensus.io/trace"
|
|
)
|
|
|
|
// ListTrustedPeer retrieves data about the node's trusted peers.
|
|
func (s *Server) ListTrustedPeer(w http.ResponseWriter, r *http.Request) {
|
|
_, span := trace.StartSpan(r.Context(), "node.ListTrustedPeer")
|
|
defer span.End()
|
|
|
|
peerStatus := s.PeersFetcher.Peers()
|
|
allIds := s.PeersFetcher.Peers().GetTrustedPeers()
|
|
allPeers := make([]*Peer, 0, len(allIds))
|
|
for _, id := range allIds {
|
|
p, err := httpPeerInfo(peerStatus, id)
|
|
if err != nil {
|
|
errJson := &httputil.DefaultJsonError{
|
|
Message: errors.Wrapf(err, "Could not get peer info").Error(),
|
|
Code: http.StatusInternalServerError,
|
|
}
|
|
httputil.WriteError(w, errJson)
|
|
return
|
|
}
|
|
// peers added into trusted set but never connected should also be listed
|
|
if p == nil {
|
|
p = &Peer{
|
|
PeerID: id.String(),
|
|
Enr: "",
|
|
LastSeenP2PAddress: "",
|
|
State: eth.ConnectionState(corenet.NotConnected).String(),
|
|
Direction: eth.PeerDirection(corenet.DirUnknown).String(),
|
|
}
|
|
}
|
|
allPeers = append(allPeers, p)
|
|
}
|
|
response := &PeersResponse{Peers: allPeers}
|
|
httputil.WriteJson(w, response)
|
|
}
|
|
|
|
// AddTrustedPeer adds a new peer into node's trusted peer set by Multiaddr
|
|
func (s *Server) AddTrustedPeer(w http.ResponseWriter, r *http.Request) {
|
|
_, span := trace.StartSpan(r.Context(), "node.AddTrustedPeer")
|
|
defer span.End()
|
|
|
|
body, err := io.ReadAll(r.Body)
|
|
if err != nil {
|
|
errJson := &httputil.DefaultJsonError{
|
|
Message: errors.Wrapf(err, "Could not read request body").Error(),
|
|
Code: http.StatusInternalServerError,
|
|
}
|
|
httputil.WriteError(w, errJson)
|
|
return
|
|
}
|
|
var addrRequest *AddrRequest
|
|
err = json.Unmarshal(body, &addrRequest)
|
|
if err != nil {
|
|
errJson := &httputil.DefaultJsonError{
|
|
Message: errors.Wrapf(err, "Could not decode request body into peer address").Error(),
|
|
Code: http.StatusBadRequest,
|
|
}
|
|
httputil.WriteError(w, errJson)
|
|
return
|
|
}
|
|
info, err := peer.AddrInfoFromString(addrRequest.Addr)
|
|
if err != nil {
|
|
errJson := &httputil.DefaultJsonError{
|
|
Message: errors.Wrapf(err, "Could not derive peer info from multiaddress").Error(),
|
|
Code: http.StatusBadRequest,
|
|
}
|
|
httputil.WriteError(w, errJson)
|
|
return
|
|
}
|
|
|
|
// also add new peerdata to peers
|
|
direction, err := s.PeersFetcher.Peers().Direction(info.ID)
|
|
if err != nil {
|
|
s.PeersFetcher.Peers().Add(nil, info.ID, info.Addrs[0], corenet.DirUnknown)
|
|
} else {
|
|
s.PeersFetcher.Peers().Add(nil, info.ID, info.Addrs[0], direction)
|
|
}
|
|
|
|
var ids []peer.ID
|
|
ids = append(ids, info.ID)
|
|
s.PeersFetcher.Peers().SetTrustedPeers(ids)
|
|
w.WriteHeader(http.StatusOK)
|
|
}
|
|
|
|
// RemoveTrustedPeer removes peer from our trusted peer set but does not close connection.
|
|
func (s *Server) RemoveTrustedPeer(w http.ResponseWriter, r *http.Request) {
|
|
_, span := trace.StartSpan(r.Context(), "node.RemoveTrustedPeer")
|
|
defer span.End()
|
|
|
|
segments := strings.Split(r.URL.Path, "/")
|
|
id := segments[len(segments)-1]
|
|
peerId, err := peer.Decode(id)
|
|
if err != nil {
|
|
errJson := &httputil.DefaultJsonError{
|
|
Message: errors.Wrapf(err, "Could not decode peer id").Error(),
|
|
Code: http.StatusBadRequest,
|
|
}
|
|
httputil.WriteError(w, errJson)
|
|
return
|
|
}
|
|
|
|
// if the peer is not a trusted peer, do nothing but return 200
|
|
if !s.PeersFetcher.Peers().IsTrustedPeers(peerId) {
|
|
w.WriteHeader(http.StatusOK)
|
|
return
|
|
}
|
|
|
|
var ids []peer.ID
|
|
ids = append(ids, peerId)
|
|
s.PeersFetcher.Peers().DeleteTrustedPeers(ids)
|
|
w.WriteHeader(http.StatusOK)
|
|
}
|
|
|
|
// httpPeerInfo does the same thing as peerInfo function in node.go but returns the
|
|
// http peer response.
|
|
func httpPeerInfo(peerStatus *peers.Status, id peer.ID) (*Peer, error) {
|
|
enr, err := peerStatus.ENR(id)
|
|
if err != nil {
|
|
if errors.Is(err, peerdata.ErrPeerUnknown) {
|
|
return nil, nil
|
|
}
|
|
return nil, errors.Wrap(err, "could not obtain ENR")
|
|
}
|
|
var serializedEnr string
|
|
if enr != nil {
|
|
serializedEnr, err = p2p.SerializeENR(enr)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not serialize ENR")
|
|
}
|
|
}
|
|
address, err := peerStatus.Address(id)
|
|
if err != nil {
|
|
if errors.Is(err, peerdata.ErrPeerUnknown) {
|
|
return nil, nil
|
|
}
|
|
return nil, errors.Wrap(err, "could not obtain address")
|
|
}
|
|
connectionState, err := peerStatus.ConnectionState(id)
|
|
if err != nil {
|
|
if errors.Is(err, peerdata.ErrPeerUnknown) {
|
|
return nil, nil
|
|
}
|
|
return nil, errors.Wrap(err, "could not obtain connection state")
|
|
}
|
|
direction, err := peerStatus.Direction(id)
|
|
if err != nil {
|
|
if errors.Is(err, peerdata.ErrPeerUnknown) {
|
|
return nil, nil
|
|
}
|
|
return nil, errors.Wrap(err, "could not obtain direction")
|
|
}
|
|
if eth.PeerDirection(direction) == eth.PeerDirection_UNKNOWN {
|
|
return nil, nil
|
|
}
|
|
v1ConnState := eth.ConnectionState(connectionState).String()
|
|
v1PeerDirection := eth.PeerDirection(direction).String()
|
|
p := Peer{
|
|
PeerID: id.String(),
|
|
State: v1ConnState,
|
|
Direction: v1PeerDirection,
|
|
}
|
|
if address != nil {
|
|
p.LastSeenP2PAddress = address.String()
|
|
}
|
|
if serializedEnr != "" {
|
|
p.Enr = "enr:" + serializedEnr
|
|
}
|
|
|
|
return &p, nil
|
|
}
|