Newer
Older
"github.com/bio-routing/bio-rd/routingtable/filter"
"github.com/bio-routing/bio-rd/net"
"github.com/bio-routing/bio-rd/route"
"github.com/bio-routing/bio-rd/routingtable"
)
// AdjRIBIn represents an Adjacency RIB In as described in RFC4271
type AdjRIBIn struct {
routingtable.ClientManager
Maximilian Wilhelm
committed
rt *routingtable.RoutingTable
mu sync.RWMutex
exportFilter *filter.Filter
contributingASNs *routingtable.ContributingASNs
Maximilian Wilhelm
committed
routerID uint32
clusterID uint32
func New(exportFilter *filter.Filter, contributingASNs *routingtable.ContributingASNs, routerID uint32, clusterID uint32, addPathRX bool) *AdjRIBIn {
a := &AdjRIBIn{
Maximilian Wilhelm
committed
rt: routingtable.NewRoutingTable(),
exportFilter: exportFilter,
contributingASNs: contributingASNs,
Maximilian Wilhelm
committed
routerID: routerID,
clusterID: clusterID,
a.ClientManager = routingtable.NewClientManager(a)
return a
}
// UpdateNewClient sends current state to a new client
func (a *AdjRIBIn) UpdateNewClient(client routingtable.RouteTableClient) error {
a.mu.RLock()
defer a.mu.RUnlock()
routes := a.rt.Dump()
for _, route := range routes {
paths := route.Paths()
for _, path := range paths {
path, reject := a.exportFilter.ProcessTerms(route.Prefix(), path)
if reject {
continue
}
err := client.AddPath(route.Prefix(), path)
if err != nil {
log.WithField("Sender", "AdjRIBOutAddPath").WithError(err).Error("Could not send update to client")
}
// RouteCount returns the number of stored routes
func (a *AdjRIBIn) RouteCount() int64 {
return a.rt.GetRouteCount()
}
// AddPath replaces the path for prefix `pfx`. If the prefix doesn't exist it is added.
func (a *AdjRIBIn) AddPath(pfx net.Prefix, p *route.Path) error {
Maximilian Wilhelm
committed
// RFC4456 Sect. 8: Ignore route with our RouterID as OriginatorID
if p.BGPPath.OriginatorID == a.routerID {
Maximilian Wilhelm
committed
return nil
}
// RFC4456 Sect. 8: Ignore routes which contian our ClusterID in their ClusterList
if len(p.BGPPath.ClusterList) > 0 {
for _, cid := range p.BGPPath.ClusterList {
if cid == a.clusterID {
return nil
}
}
}
if a.addPathRX {
a.rt.AddPath(pfx, p)
} else {
oldPaths := a.rt.ReplacePath(pfx, p)
a.removePathsFromClients(pfx, oldPaths)
}
p, reject := a.exportFilter.ProcessTerms(pfx, p)
if reject {
return nil
}
Maximilian Wilhelm
committed
// Bail out - for all clients for now - if any of our ASNs is within the path
if a.ourASNsInPath(p) {
return nil
}
for _, client := range a.ClientManager.Clients() {
client.AddPath(pfx, p)
}
Maximilian Wilhelm
committed
func (a *AdjRIBIn) ourASNsInPath(p *route.Path) bool {
Maximilian Wilhelm
committed
for _, asn := range pathSegment.ASNs {
if a.contributingASNs.IsContributingASN(asn) {
return true
}
}
}
return false
}
// RemovePath removes the path for prefix `pfx`
func (a *AdjRIBIn) RemovePath(pfx net.Prefix, p *route.Path) bool {
return false
oldPaths := r.Paths()
for _, path := range oldPaths {
if a.addPathRX {
if path.BGPPath.PathIdentifier != p.BGPPath.PathIdentifier {
continue
}
}
return true
func (a *AdjRIBIn) removePathsFromClients(pfx net.Prefix, paths []*route.Path) {
for _, path := range paths {
path, reject := a.exportFilter.ProcessTerms(pfx, path)
if reject {
continue
}
for _, client := range a.ClientManager.Clients() {
client.RemovePath(pfx, path)
}
}
}
func (a *AdjRIBIn) RT() *routingtable.RoutingTable {
return a.rt
}