Skip to content

Commit

Permalink
bgp: support multiple peer addresses
Browse files Browse the repository at this point in the history
Signed-off-by: 张祖建 <[email protected]>
  • Loading branch information
zhangzujian committed Oct 8, 2023
1 parent c493a95 commit 0d3402a
Show file tree
Hide file tree
Showing 2 changed files with 108 additions and 93 deletions.
127 changes: 67 additions & 60 deletions pkg/speaker/config.go
Original file line number Diff line number Diff line change
Expand Up @@ -40,8 +40,8 @@ type Configuration struct {
GrpcPort uint32
ClusterAs uint32
RouterID string
NeighborAddress string
NeighborIPv6Address string
NeighborAddresses []string
NeighborIPv6Addresses []string
NeighborAs uint32
AuthPassword string
HoldTime float64
Expand Down Expand Up @@ -71,8 +71,8 @@ func ParseFlags() (*Configuration, error) {
argGrpcPort = pflag.Uint32("grpc-port", DefaultBGPGrpcPort, "The port for grpc to listen, default:50051")
argClusterAs = pflag.Uint32("cluster-as", DefaultBGPClusterAs, "The as number of container network, default 65000")
argRouterID = pflag.String("router-id", "", "The address for the speaker to use as router id, default the node ip")
argNeighborAddress = pflag.String("neighbor-address", "", "The router address the speaker connects to.")
argNeighborIPv6Address = pflag.String("neighbor-ipv6-address", "", "The router address the speaker connects to.")
argNeighborAddress = pflag.String("neighbor-address", "", "Comma separated IPv4 router addresses the speaker connects to.")
argNeighborIPv6Address = pflag.String("neighbor-ipv6-address", "", "Comma separated IPv6 router addresses the speaker connects to.")
argNeighborAs = pflag.Uint32("neighbor-as", DefaultBGPNeighborAs, "The router as number, default 65001")
argAuthPassword = pflag.String("auth-password", "", "bgp peer auth password")
argHoldTime = pflag.Duration("holdtime", DefaultBGPHoldtime, "ovn-speaker goes down abnormally, the local saving time of BGP route will be affected.Holdtime must be in the range 3s to 65536s. (default 90s)")
Expand Down Expand Up @@ -107,12 +107,6 @@ func ParseFlags() (*Configuration, error) {
if *argRouterID != "" && net.ParseIP(*argRouterID) == nil {
return nil, fmt.Errorf("invalid router-id format: %s", *argRouterID)
}
if *argNeighborAddress != "" && net.ParseIP(*argNeighborAddress).To4() == nil {
return nil, fmt.Errorf("invalid neighbor-address format: %s", *argNeighborAddress)
}
if *argNeighborIPv6Address != "" && net.ParseIP(*argNeighborIPv6Address).To16() == nil {
return nil, fmt.Errorf("invalid neighbor-ipv6-address format: %s", *argNeighborIPv6Address)
}
if *argEbgpMultihopTTL < 1 || *argEbgpMultihopTTL > 255 {
return nil, errors.New("the bgp MultihopTtl must be in the range 1 to 255")
}
Expand All @@ -123,8 +117,6 @@ func ParseFlags() (*Configuration, error) {
GrpcPort: *argGrpcPort,
ClusterAs: *argClusterAs,
RouterID: *argRouterID,
NeighborAddress: *argNeighborAddress,
NeighborIPv6Address: *argNeighborIPv6Address,
NeighborAs: *argNeighborAs,
AuthPassword: *argAuthPassword,
HoldTime: ht,
Expand All @@ -138,6 +130,23 @@ func ParseFlags() (*Configuration, error) {
EbgpMultihopTTL: *argEbgpMultihopTTL,
}

if *argNeighborAddress != "" {
config.NeighborAddresses = strings.Split(*argNeighborAddress, ",")
for _, addr := range config.NeighborAddresses {
if ip := net.ParseIP(addr); ip == nil || ip.To4() == nil {
return nil, fmt.Errorf("invalid neighbor-address format: %s", *argNeighborAddress)
}
}
}
if *argNeighborIPv6Address != "" {
config.NeighborIPv6Addresses = strings.Split(*argNeighborIPv6Address, ",")
for _, addr := range config.NeighborIPv6Addresses {
if ip := net.ParseIP(addr); ip == nil || ip.To16() == nil {
return nil, fmt.Errorf("invalid neighbor-ipv6-address format: %s", *argNeighborIPv6Address)
}
}
}

if config.RouterID == "" {
config.RouterID = os.Getenv("POD_IP")
if config.RouterID == "" {
Expand Down Expand Up @@ -207,19 +216,16 @@ func (config *Configuration) checkGracefulRestartOptions() error {

func (config *Configuration) initBgpServer() error {
maxSize := 256 << 20
peersMap := make(map[api.Family_Afi]string)
var listenPort int32 = -1
grpcOpts := []grpc.ServerOption{grpc.MaxRecvMsgSize(maxSize), grpc.MaxSendMsgSize(maxSize)}
s := gobgp.NewBgpServer(
gobgp.GrpcListenAddress(fmt.Sprintf("%s:%d", config.GrpcHost, config.GrpcPort)),
gobgp.GrpcOption(grpcOpts))
go s.Serve()

if config.NeighborAddress != "" {
peersMap[api.Family_AFI_IP] = config.NeighborAddress
}
if config.NeighborIPv6Address != "" {
peersMap[api.Family_AFI_IP6] = config.NeighborIPv6Address
peersMap := map[api.Family_Afi][]string{
api.Family_AFI_IP: config.NeighborAddresses,
api.Family_AFI_IP6: config.NeighborIPv6Addresses,
}

if config.PassiveMode {
Expand All @@ -235,56 +241,57 @@ func (config *Configuration) initBgpServer() error {
}); err != nil {
return err
}
for ipFamily, address := range peersMap {
peer := &api.Peer{
Timers: &api.Timers{Config: &api.TimersConfig{HoldTime: uint64(config.HoldTime)}},
Conf: &api.PeerConf{
NeighborAddress: address,
PeerAsn: config.NeighborAs,
},
Transport: &api.Transport{
PassiveMode: config.PassiveMode,
},
}
if config.EbgpMultihopTTL != DefaultEbgpMultiHop {
peer.EbgpMultihop = &api.EbgpMultihop{
Enabled: true,
MultihopTtl: uint32(config.EbgpMultihopTTL),
for ipFamily, addresses := range peersMap {
for _, addr := range addresses {
peer := &api.Peer{
Timers: &api.Timers{Config: &api.TimersConfig{HoldTime: uint64(config.HoldTime)}},
Conf: &api.PeerConf{
NeighborAddress: addr,
PeerAsn: config.NeighborAs,
},
Transport: &api.Transport{
PassiveMode: config.PassiveMode,
},
}
}
if config.AuthPassword != "" {
peer.Conf.AuthPassword = config.AuthPassword
}
if config.GracefulRestart {

if err := config.checkGracefulRestartOptions(); err != nil {
return err
if config.EbgpMultihopTTL != DefaultEbgpMultiHop {
peer.EbgpMultihop = &api.EbgpMultihop{
Enabled: true,
MultihopTtl: uint32(config.EbgpMultihopTTL),
}
}
peer.GracefulRestart = &api.GracefulRestart{
Enabled: true,
RestartTime: uint32(config.GracefulRestartTime.Seconds()),
DeferralTime: uint32(config.GracefulRestartDeferralTime.Seconds()),
LocalRestarting: true,
if config.AuthPassword != "" {
peer.Conf.AuthPassword = config.AuthPassword
}
peer.AfiSafis = []*api.AfiSafi{
{
Config: &api.AfiSafiConfig{
Family: &api.Family{Afi: ipFamily, Safi: api.Family_SAFI_UNICAST},
Enabled: true,
},
MpGracefulRestart: &api.MpGracefulRestart{
Config: &api.MpGracefulRestartConfig{
if config.GracefulRestart {
if err := config.checkGracefulRestartOptions(); err != nil {
return err
}
peer.GracefulRestart = &api.GracefulRestart{
Enabled: true,
RestartTime: uint32(config.GracefulRestartTime.Seconds()),
DeferralTime: uint32(config.GracefulRestartDeferralTime.Seconds()),
LocalRestarting: true,
}
peer.AfiSafis = []*api.AfiSafi{
{
Config: &api.AfiSafiConfig{
Family: &api.Family{Afi: ipFamily, Safi: api.Family_SAFI_UNICAST},
Enabled: true,
},
MpGracefulRestart: &api.MpGracefulRestart{
Config: &api.MpGracefulRestartConfig{
Enabled: true,
},
},
},
},
}
}
}

if err := s.AddPeer(context.Background(), &api.AddPeerRequest{
Peer: peer,
}); err != nil {
return err
if err := s.AddPeer(context.Background(), &api.AddPeerRequest{
Peer: peer,
}); err != nil {
return err
}
}
}

Expand Down
74 changes: 41 additions & 33 deletions pkg/speaker/subnet.go
Original file line number Diff line number Diff line change
Expand Up @@ -160,7 +160,7 @@ func (c *Controller) syncSubnetRoutes() {
}
}

if c.config.NeighborAddress != "" {
if len(c.config.NeighborAddresses) != 0 {
listPathRequest := &bgpapi.ListPathRequest{
TableType: bgpapi.TableType_GLOBAL,
Family: &bgpapi.Family{Afi: bgpapi.Family_AFI_IP, Safi: bgpapi.Family_SAFI_UNICAST},
Expand All @@ -187,8 +187,7 @@ func (c *Controller) syncSubnetRoutes() {
}
}

if c.config.NeighborIPv6Address != "" {

if len(c.config.NeighborIPv6Addresses) != 0 {
listIPv6PathRequest := &bgpapi.ListPathRequest{
TableType: bgpapi.TableType_GLOBAL,
Family: &bgpapi.Family{Afi: bgpapi.Family_AFI_IP6, Safi: bgpapi.Family_SAFI_UNICAST},
Expand Down Expand Up @@ -266,24 +265,26 @@ func (c *Controller) addRoute(route string) error {
if err != nil {
return err
}
_, err = c.config.BgpServer.AddPath(context.Background(), &bgpapi.AddPathRequest{
Path: &bgpapi.Path{
Family: &bgpapi.Family{Afi: routeAfi, Safi: bgpapi.Family_SAFI_UNICAST},
Nlri: nlri,
Pattrs: attrs,
},
})
if err != nil {
klog.Errorf("add path failed, %v", err)
return err
for _, attr := range attrs {
_, err = c.config.BgpServer.AddPath(context.Background(), &bgpapi.AddPathRequest{
Path: &bgpapi.Path{
Family: &bgpapi.Family{Afi: routeAfi, Safi: bgpapi.Family_SAFI_UNICAST},
Nlri: nlri,
Pattrs: attr,
},
})
if err != nil {
klog.Errorf("add path failed, %v", err)
return err
}
}
return nil
}

func (c *Controller) getNlriAndAttrs(route string) (*anypb.Any, []*anypb.Any, error) {
neighborAddr := c.config.NeighborAddress
func (c *Controller) getNlriAndAttrs(route string) (*anypb.Any, [][]*anypb.Any, error) {
neighborAddresses := c.config.NeighborAddresses
if util.CheckProtocol(route) == kubeovnv1.ProtocolIPv6 {
neighborAddr = c.config.NeighborIPv6Address
neighborAddresses = c.config.NeighborIPv6Addresses
}

prefix, prefixLen, err := parseRoute(route)
Expand All @@ -294,13 +295,18 @@ func (c *Controller) getNlriAndAttrs(route string) (*anypb.Any, []*anypb.Any, er
Prefix: prefix,
PrefixLen: prefixLen,
})
a1, _ := anypb.New(&bgpapi.OriginAttribute{
Origin: 0,
})
a2, _ := anypb.New(&bgpapi.NextHopAttribute{
NextHop: getNextHopAttribute(neighborAddr, c.config.RouterID),
})
attrs := []*anypb.Any{a1, a2}

attrs := make([][]*anypb.Any, 0, len(neighborAddresses))
for _, addr := range neighborAddresses {
a1, _ := anypb.New(&bgpapi.OriginAttribute{
Origin: 0,
})
a2, _ := anypb.New(&bgpapi.NextHopAttribute{
NextHop: getNextHopAttribute(addr, c.config.RouterID),
})
attrs = append(attrs, []*anypb.Any{a1, a2})
}

return nlri, attrs, err
}

Expand All @@ -314,16 +320,18 @@ func (c *Controller) delRoute(route string) error {
if err != nil {
return err
}
err = c.config.BgpServer.DeletePath(context.Background(), &bgpapi.DeletePathRequest{
Path: &bgpapi.Path{
Family: &bgpapi.Family{Afi: routeAfi, Safi: bgpapi.Family_SAFI_UNICAST},
Nlri: nlri,
Pattrs: attrs,
},
})
if err != nil {
klog.Errorf("del path failed, %v", err)
return err
for _, attr := range attrs {
err = c.config.BgpServer.DeletePath(context.Background(), &bgpapi.DeletePathRequest{
Path: &bgpapi.Path{
Family: &bgpapi.Family{Afi: routeAfi, Safi: bgpapi.Family_SAFI_UNICAST},
Nlri: nlri,
Pattrs: attr,
},
})
if err != nil {
klog.Errorf("del path failed, %v", err)
return err
}
}
return nil
}
Expand Down

0 comments on commit 0d3402a

Please sign in to comment.