mirror of
https://github.com/osrg/gobgp.git
synced 2024-05-11 05:55:10 +00:00
263 lines
6.6 KiB
Go
263 lines
6.6 KiB
Go
// Copyright (C) 2014 Nippon Telegraph and Telephone Corporation.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
|
// implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package table
|
|
|
|
import (
|
|
log "github.com/Sirupsen/logrus"
|
|
"github.com/armon/go-radix"
|
|
"github.com/osrg/gobgp/packet/bgp"
|
|
"sort"
|
|
)
|
|
|
|
type Table struct {
|
|
routeFamily bgp.RouteFamily
|
|
destinations map[string]*Destination
|
|
}
|
|
|
|
func NewTable(rf bgp.RouteFamily) *Table {
|
|
return &Table{
|
|
routeFamily: rf,
|
|
destinations: make(map[string]*Destination),
|
|
}
|
|
}
|
|
|
|
func (t *Table) GetRoutefamily() bgp.RouteFamily {
|
|
return t.routeFamily
|
|
}
|
|
|
|
func (t *Table) insert(path *Path) *Destination {
|
|
t.validatePath(path)
|
|
dest := t.getOrCreateDest(path.GetNlri())
|
|
|
|
if path.IsWithdraw {
|
|
// withdraw insert
|
|
dest.addWithdraw(path)
|
|
} else {
|
|
// path insert
|
|
dest.addNewPath(path)
|
|
}
|
|
return dest
|
|
}
|
|
|
|
func (t *Table) DeleteDestByPeer(peerInfo *PeerInfo) []*Destination {
|
|
dsts := []*Destination{}
|
|
for _, dst := range t.destinations {
|
|
match := false
|
|
for _, p := range dst.knownPathList {
|
|
if p.GetSource().Equal(peerInfo) {
|
|
dst.addWithdraw(p)
|
|
match = true
|
|
}
|
|
}
|
|
if match {
|
|
dsts = append(dsts, dst)
|
|
}
|
|
}
|
|
return dsts
|
|
}
|
|
|
|
func (t *Table) deletePathsByVrf(vrf *Vrf) []*Path {
|
|
pathList := make([]*Path, 0)
|
|
for _, dest := range t.destinations {
|
|
for _, p := range dest.knownPathList {
|
|
var rd bgp.RouteDistinguisherInterface
|
|
nlri := p.GetNlri()
|
|
switch nlri.(type) {
|
|
case *bgp.LabeledVPNIPAddrPrefix:
|
|
rd = nlri.(*bgp.LabeledVPNIPAddrPrefix).RD
|
|
case *bgp.LabeledVPNIPv6AddrPrefix:
|
|
rd = nlri.(*bgp.LabeledVPNIPv6AddrPrefix).RD
|
|
case *bgp.EVPNNLRI:
|
|
rd = nlri.(*bgp.EVPNNLRI).RD()
|
|
default:
|
|
return pathList
|
|
}
|
|
if p.IsLocal() && vrf.Rd.String() == rd.String() {
|
|
pathList = append(pathList, p.Clone(true))
|
|
break
|
|
}
|
|
}
|
|
}
|
|
return pathList
|
|
}
|
|
|
|
func (t *Table) deleteRTCPathsByVrf(vrf *Vrf, vrfs map[string]*Vrf) []*Path {
|
|
pathList := make([]*Path, 0)
|
|
if t.routeFamily != bgp.RF_RTC_UC {
|
|
return pathList
|
|
}
|
|
for _, target := range vrf.ImportRt {
|
|
lhs := target.String()
|
|
for _, dest := range t.destinations {
|
|
nlri := dest.GetNlri().(*bgp.RouteTargetMembershipNLRI)
|
|
rhs := nlri.RouteTarget.String()
|
|
if lhs == rhs && isLastTargetUser(vrfs, target) {
|
|
for _, p := range dest.knownPathList {
|
|
if p.IsLocal() {
|
|
pathList = append(pathList, p.Clone(true))
|
|
break
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return pathList
|
|
}
|
|
|
|
func (t *Table) deleteDestByNlri(nlri bgp.AddrPrefixInterface) *Destination {
|
|
destinations := t.GetDestinations()
|
|
dest := destinations[t.tableKey(nlri)]
|
|
if dest != nil {
|
|
delete(destinations, t.tableKey(nlri))
|
|
if len(destinations) == 0 {
|
|
t.destinations = make(map[string]*Destination)
|
|
}
|
|
}
|
|
return dest
|
|
}
|
|
|
|
func (t *Table) deleteDest(dest *Destination) {
|
|
destinations := t.GetDestinations()
|
|
delete(destinations, t.tableKey(dest.GetNlri()))
|
|
if len(destinations) == 0 {
|
|
t.destinations = make(map[string]*Destination)
|
|
}
|
|
}
|
|
|
|
func (t *Table) validatePath(path *Path) {
|
|
if path == nil {
|
|
log.WithFields(log.Fields{
|
|
"Topic": "Table",
|
|
"Key": t.routeFamily,
|
|
}).Error("path is nil")
|
|
}
|
|
if path.GetRouteFamily() != t.routeFamily {
|
|
log.WithFields(log.Fields{
|
|
"Topic": "Table",
|
|
"Key": t.routeFamily,
|
|
"Prefix": path.GetNlri().String(),
|
|
"ReceivedRf": path.GetRouteFamily().String(),
|
|
}).Error("Invalid path. RouteFamily mismatch")
|
|
}
|
|
if attr := path.getPathAttr(bgp.BGP_ATTR_TYPE_AS_PATH); attr != nil {
|
|
pathParam := attr.(*bgp.PathAttributeAsPath).Value
|
|
for _, as := range pathParam {
|
|
_, y := as.(*bgp.As4PathParam)
|
|
if !y {
|
|
log.WithFields(log.Fields{
|
|
"Topic": "Table",
|
|
"Key": t.routeFamily,
|
|
"As": as,
|
|
}).Fatal("AsPathParam must be converted to As4PathParam")
|
|
}
|
|
}
|
|
}
|
|
if attr := path.getPathAttr(bgp.BGP_ATTR_TYPE_AS4_PATH); attr != nil {
|
|
log.WithFields(log.Fields{
|
|
"Topic": "Table",
|
|
"Key": t.routeFamily,
|
|
}).Fatal("AS4_PATH must be converted to AS_PATH")
|
|
}
|
|
if path.GetNlri() == nil {
|
|
log.WithFields(log.Fields{
|
|
"Topic": "Table",
|
|
"Key": t.routeFamily,
|
|
}).Fatal("path's nlri is nil")
|
|
}
|
|
}
|
|
|
|
func (t *Table) getOrCreateDest(nlri bgp.AddrPrefixInterface) *Destination {
|
|
tableKey := t.tableKey(nlri)
|
|
dest := t.GetDestination(tableKey)
|
|
// If destination for given prefix does not exist we create it.
|
|
if dest == nil {
|
|
log.WithFields(log.Fields{
|
|
"Topic": "Table",
|
|
"Key": tableKey,
|
|
}).Debugf("create Destination")
|
|
dest = NewDestination(nlri)
|
|
t.setDestination(tableKey, dest)
|
|
}
|
|
return dest
|
|
}
|
|
|
|
func (t *Table) GetSortedDestinations() []*Destination {
|
|
results := make([]*Destination, 0, len(t.GetDestinations()))
|
|
switch t.routeFamily {
|
|
case bgp.RF_IPv4_UC, bgp.RF_IPv6_UC:
|
|
r := radix.New()
|
|
for _, dst := range t.GetDestinations() {
|
|
r.Insert(dst.RadixKey, dst)
|
|
}
|
|
r.Walk(func(s string, v interface{}) bool {
|
|
results = append(results, v.(*Destination))
|
|
return false
|
|
})
|
|
case bgp.RF_FS_IPv4_UC, bgp.RF_FS_IPv6_UC, bgp.RF_FS_IPv4_VPN, bgp.RF_FS_IPv6_VPN, bgp.RF_FS_L2_VPN:
|
|
for _, dst := range t.GetDestinations() {
|
|
results = append(results, dst)
|
|
}
|
|
sort.Sort(destinations(results))
|
|
default:
|
|
for _, dst := range t.GetDestinations() {
|
|
results = append(results, dst)
|
|
}
|
|
}
|
|
return results
|
|
}
|
|
|
|
func (t *Table) GetDestinations() map[string]*Destination {
|
|
return t.destinations
|
|
}
|
|
func (t *Table) setDestinations(destinations map[string]*Destination) {
|
|
t.destinations = destinations
|
|
}
|
|
func (t *Table) GetDestination(key string) *Destination {
|
|
dest, ok := t.destinations[key]
|
|
if ok {
|
|
return dest
|
|
} else {
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func (t *Table) setDestination(key string, dest *Destination) {
|
|
t.destinations[key] = dest
|
|
}
|
|
|
|
func (t *Table) tableKey(nlri bgp.AddrPrefixInterface) string {
|
|
return nlri.String()
|
|
}
|
|
|
|
func (t *Table) Bests(id string) []*Path {
|
|
paths := make([]*Path, 0, len(t.destinations))
|
|
for _, dst := range t.destinations {
|
|
path := dst.GetBestPath(id)
|
|
if path != nil {
|
|
paths = append(paths, path)
|
|
}
|
|
}
|
|
return paths
|
|
}
|
|
|
|
func (t *Table) GetKnownPathList(id string) []*Path {
|
|
paths := make([]*Path, 0, len(t.destinations))
|
|
for _, dst := range t.destinations {
|
|
paths = append(paths, dst.GetKnownPathList(id)...)
|
|
}
|
|
return paths
|
|
}
|