Skip to content

Commit

Permalink
Add method to get BMP Neighbors as API Session objects
Browse files Browse the repository at this point in the history
This method can be used to expose a BGPServer-like API with a BMP server
as backend.
  • Loading branch information
sebageek committed May 24, 2020
1 parent 6f77acc commit a702495
Show file tree
Hide file tree
Showing 2 changed files with 65 additions and 0 deletions.
43 changes: 43 additions & 0 deletions protocols/bgp/server/bmp_router.go
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ import (
"github.com/bio-routing/bio-rd/routingtable/filter"
"github.com/bio-routing/bio-rd/routingtable/vrf"
"github.com/bio-routing/tflow2/convert"
"github.com/bio-routing/bio-rd/protocols/bgp/api"
"github.com/bio-routing/bio-rd/routingtable/adjRIBIn"
"github.com/bio-routing/bio-rd/routingtable/adjRIBOut"
)
Expand Down Expand Up @@ -128,6 +129,48 @@ func (r *Router) serve(con net.Conn) {
}
}

// GetNeighborSessions returns all neighbors as API session objects
func (r *Router) GetNeighborSessions() []*api.Session {
sessions := make([]*api.Session, 0)

for _, neigh := range r.neighborManager.list() {
estSince := neigh.fsm.establishedTime.Unix()
if estSince < 0 {
estSince = 0
}

// for now get this from adjRibIn/adjRibOut, can be replaced when we
// bmp gets its own bgpSrv or Router gets the bmpMetricsService
var routesReceived, routesSent uint64
for _, afi := range []uint16{packet.IPv4AFI, packet.IPv6AFI} {
ribIn, err1 := r.GetNeighborRIBIn(neigh.fsm.peer.addr, afi, packet.UnicastSAFI)
if err1 == nil {
routesReceived += uint64(ribIn.RouteCount())
}

// adjRIBOut might not work properly with BMP, keeping it here for when it will
ribOut, err2 := r.GetNeighborRIBOut(neigh.fsm.peer.addr, afi, packet.UnicastSAFI)
if err2 == nil {
routesSent += uint64(ribOut.RouteCount())
}
}
session := &api.Session{
LocalAddress: neigh.fsm.peer.localAddr.ToProto(),
NeighborAddress: neigh.fsm.peer.addr.ToProto(),
LocalAsn: neigh.localAS,
PeerAsn: neigh.peerAS,
Status: stateToProto(neigh.fsm.state),
Stats: &api.SessionStats{
RoutesReceived: routesReceived,
RoutesExported: routesSent,
},
EstablishedSince: uint64(estSince),
}
sessions = append(sessions, session)
}
return sessions
}

func (r *Router) processMsg(msg []byte) {
bmpMsg, err := bmppkt.Decode(msg)
if err != nil {
Expand Down
22 changes: 22 additions & 0 deletions protocols/bgp/server/fsm.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ import (
"time"

"github.com/bio-routing/bio-rd/net/tcp"
"github.com/bio-routing/bio-rd/protocols/bgp/api"
"github.com/bio-routing/bio-rd/protocols/bgp/packet"
"github.com/bio-routing/bio-rd/routingtable/filter"
"github.com/pkg/errors"
Expand Down Expand Up @@ -239,6 +240,27 @@ func stateName(s state) string {
}
}

func stateToProto(s state) api.Session_State {
switch s.(type) {
case *idleState:
return api.Session_Idle
case *connectState:
return api.Session_Connect
case *activeState:
return api.Session_Active
case *openSentState:
return api.Session_OpenSent
case *openConfirmState:
return api.Session_OpenConfirmed
case *establishedState:
return api.Session_Established
case *ceaseState:
return api.Session_Active // substitution
default:
panic(fmt.Sprintf("Unknown state: %v", s))
}
}

func (fsm *FSM) cease() {
fsm.eventCh <- Cease
}
Expand Down

0 comments on commit a702495

Please sign in to comment.