Skip to content

Commit

Permalink
fix(evpn): report error in comp details review fixes
Browse files Browse the repository at this point in the history
Signed-off-by: atulpatel261194 <Atul.Patel@intel.com>
  • Loading branch information
atulpatel261194 authored and artek-koltun committed Dec 24, 2024
1 parent ad8591e commit 12cd7c0
Show file tree
Hide file tree
Showing 2 changed files with 58 additions and 34 deletions.
48 changes: 20 additions & 28 deletions pkg/frr/frr.go
Original file line number Diff line number Diff line change
Expand Up @@ -495,8 +495,9 @@ func setUpVrf(vrf *infradb.Vrf) (string, bool) {
// Configure the vrf in FRR and set up BGP EVPN for it
vrfName := fmt.Sprintf("vrf %s", path.Base(vrf.Name))
vniID := fmt.Sprintf("vni %s", strconv.Itoa(int(*vrf.Spec.Vni)))
data, err := Frr.FrrZebraCmd(ctx, fmt.Sprintf("configure terminal\n %s\n %s\n exit-vrf\n exit", vrfName, vniID))
if err != nil || checkFrrResult(data, false) {

data, err := Frr.FrrZebraCmd(ctx, fmt.Sprintf("configure terminal\n %s\n %s\n exit-vrf\n exit", vrfName, vniID), false)
if err != nil {
log.Printf("FRR: Error Executing frr config t %s %s exit-vrf exit \n Error: is %v data is %v\n", vrfName, vniID, err.Error(), data)
return fmt.Sprintf("FRR: Error Executing frr config t %s %s exit-vrf exit data %v \n Error: %v Data: %v\n", vrfName, vniID, data, err, data), false
}
Expand All @@ -512,8 +513,8 @@ func setUpVrf(vrf *infradb.Vrf) (string, bool) {
} else {
LbiP = fmt.Sprintf("%+v", vrf.Spec.LoopbackIP.IP)
}
data, err = Frr.FrrBgpCmd(ctx, fmt.Sprintf("configure terminal\n router bgp %+v vrf %s\n bgp router-id %s\n no bgp ebgp-requires-policy\n no bgp hard-administrative-reset\n no bgp graceful-restart notification\n address-family ipv4 unicast\n redistribute connected\n redistribute static\n exit-address-family\n address-family l2vpn evpn\n advertise ipv4 unicast\n exit-address-family\n exit", localas, path.Base(vrf.Name), LbiP))
if err != nil || checkFrrResult(data, false) {
data, err = Frr.FrrBgpCmd(ctx, fmt.Sprintf("configure terminal\n router bgp %+v vrf %s\n bgp router-id %s\n no bgp ebgp-requires-policy\n no bgp hard-administrative-reset\n no bgp graceful-restart notification\n address-family ipv4 unicast\n redistribute connected\n redistribute static\n exit-address-family\n address-family l2vpn evpn\n advertise ipv4 unicast\n exit-address-family\n exit", localas, path.Base(vrf.Name), LbiP), false)
if err != nil {
log.Printf("FRR: Error Executing config t bgpVrfName router bgp %+v vrf %s bgp_route_id %s no bgp ebgp-requires-policy exit-vrf exit data %v \n", localas, vrf.Name, LbiP, data)
return fmt.Sprintf("FRR: Error Executing config t bgpVrfName router bgp %+v vrf %s bgp_route_id %s no bgp ebgp-requires-policy exit-vrf exit data %v \n", localas, vrf.Name, LbiP, data), false
}
Expand All @@ -524,8 +525,8 @@ func setUpVrf(vrf *infradb.Vrf) (string, bool) {
log.Printf("FRR: Executed config t bgpVrfName router bgp %+v vrf %s bgp_route_id %s no bgp ebgp-requires-policy exit-vrf exit\n", localas, vrf.Name, LbiP)
// Update the vrf with attributes from FRR
cmd := fmt.Sprintf("show bgp l2vpn evpn vni %d json", *vrf.Spec.Vni)
cp, err := Frr.FrrBgpCmd(ctx, cmd)
if err != nil || checkFrrResult(cp, true) {
cp, err := Frr.FrrBgpCmd(ctx, cmd, true)
if err != nil {
log.Printf("FRR Error-show bgp l2vpn evpn vni %v cp %v", err, cp)
}
err = Frr.Save(ctx)
Expand All @@ -548,8 +549,8 @@ func setUpVrf(vrf *infradb.Vrf) (string, bool) {
log.Printf("error-%v", err)
}
cmd = fmt.Sprintf("show bgp vrf %s json", path.Base(vrf.Name))
cp, err = Frr.FrrBgpCmd(ctx, cmd)
if err != nil || checkFrrResult(cp, true) {
cp, err = Frr.FrrBgpCmd(ctx, cmd, true)
if err != nil {
log.Printf("error-%v", err)
}
err = Frr.Save(ctx)
Expand Down Expand Up @@ -579,11 +580,6 @@ func setUpVrf(vrf *infradb.Vrf) (string, bool) {
return "", true
}

// checkFrrResult checks the vrf result
func checkFrrResult(cp string, show bool) bool {
return ((show && reflect.ValueOf(cp).IsZero()) || strings.Contains(cp, "warning") || strings.Contains(cp, "unknown") || strings.Contains(cp, "Unknown") || strings.Contains(cp, "Warning") || strings.Contains(cp, "Ambiguous") || strings.Contains(cp, "specified does not exist") || strings.Contains(cp, "Error"))
}

// setUpSvi sets up the svi
func setUpSvi(svi *infradb.Svi) (string, bool) {
BrObj, err := infradb.GetLB(svi.Spec.LogicalBridge)
Expand All @@ -604,9 +600,9 @@ func setUpSvi(svi *infradb.Svi) (string, bool) {
neighlinkSr := fmt.Sprintf("neighbor %s soft-reconfiguration inbound\n", linkSvi)
bgpListen := fmt.Sprintf(" bgp listen range %s peer-group %s\n", svi.Spec.GatewayIPs[0], linkSvi)

data, err := Frr.FrrBgpCmd(ctx, fmt.Sprintf("configure terminal\n %s bgp disable-ebgp-connected-route-check\n %s %s %s %s %s %s exit", bgpVrfName, neighlink, neighlinkRe, neighlinkGw, neighlinkOv, neighlinkSr, bgpListen))
data, err := Frr.FrrBgpCmd(ctx, fmt.Sprintf("configure terminal\n %s bgp disable-ebgp-connected-route-check\n %s %s %s %s %s %s exit", bgpVrfName, neighlink, neighlinkRe, neighlinkGw, neighlinkOv, neighlinkSr, bgpListen), false)

if err != nil || checkFrrResult(data, false) {
if err != nil {
log.Printf("FRR: Error in conf svi %s %s command %s\n", svi.Name, path.Base(svi.Spec.Vrf), data)
return fmt.Sprintf("FRR: Error in conf svi %s %s command %s\n", svi.Name, path.Base(svi.Spec.Vrf), data), false
}
Expand All @@ -631,11 +627,10 @@ func tearDownSvi(svi *infradb.Svi) (string, bool) {
if svi.Spec.EnableBgp && !reflect.ValueOf(svi.Spec.GatewayIPs).IsZero() {
bgpVrfName := fmt.Sprintf("router bgp %+v vrf %s", localas, path.Base(svi.Spec.Vrf))
noNeigh := fmt.Sprintf("no neighbor %s peer-group", linkSvi)
data, err := Frr.FrrBgpCmd(ctx, fmt.Sprintf("configure terminal\n %s\n %s\n exit", bgpVrfName, noNeigh))
if strings.Contains(data, "Create the peer-group first") { // Trying to delete non exist peer-group return true
return "", true
}
if err != nil || checkFrrResult(data, false) {

data, err := Frr.FrrBgpCmd(ctx, fmt.Sprintf("configure terminal\n %s\n %s\n exit", bgpVrfName, noNeigh), false)

if err != nil {
log.Printf("FRR: Error in conf Delete vrf/VNI command %s\n", data)
return fmt.Sprintf("FRR: Error in conf Delete vrf/VNI command %s\n", data), false
}
Expand All @@ -656,8 +651,8 @@ func tearDownVrf(vrf *infradb.Vrf) (string, bool) {
return "", true
}

data, err := Frr.FrrZebraCmd(ctx, fmt.Sprintf("show vrf %s vni\n", path.Base(vrf.Name)))
if err != nil || checkFrrResult(data, true) {
data, err := Frr.FrrZebraCmd(ctx, fmt.Sprintf("show vrf %s vni\n", path.Base(vrf.Name)), true)
if err != nil {
log.Printf("FRR: Error %s\n", data)
return "", true
}
Expand All @@ -670,19 +665,16 @@ func tearDownVrf(vrf *infradb.Vrf) (string, bool) {
log.Printf("FRR Deleted event")
delCmd1 := fmt.Sprintf("no router bgp %+v vrf %s", localas, path.Base(vrf.Name))
delCmd2 := fmt.Sprintf("no vrf %s", path.Base(vrf.Name))
data, err = Frr.FrrBgpCmd(ctx, fmt.Sprintf("configure terminal\n %s\n exit\n", delCmd1))
if strings.Contains(data, "Can't find BGP instance") { // Trying to delete non exist VRF return true
return "", true
}
if err != nil || checkFrrResult(data, false) {
data, err = Frr.FrrBgpCmd(ctx, fmt.Sprintf("configure terminal\n %s\n exit\n", delCmd1), false)
if err != nil {
log.Printf("FRR: Error %s\n", data)
return fmt.Sprintf("FRR: Error %s\n", data), false
}
err = Frr.Save(ctx)
if err != nil {
log.Printf("FRR(tearDownVrf): Failed to run save command: %v\n", err)
}
_, err = Frr.FrrZebraCmd(ctx, fmt.Sprintf("configure terminal\n %s\n exit\n", delCmd2))
data, err = Frr.FrrZebraCmd(ctx, fmt.Sprintf("configure terminal\n %s\n exit\n", delCmd2), false)
if err != nil {
log.Printf("FRR: Error %s\n", data)
return fmt.Sprintf("FRR: Error %s\n", data), false
Expand Down
44 changes: 38 additions & 6 deletions pkg/utils/frr.go
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ package utils
import (
"bufio"
"context"
"errors"
"fmt"
"os/exec"
"strings"
Expand Down Expand Up @@ -51,8 +52,8 @@ const (
// Frr represents limited subset of functions from Frr package
type Frr interface {
TelnetDialAndCommunicate(ctx context.Context, command string, port int) (string, error)
FrrZebraCmd(ctx context.Context, command string) (string, error)
FrrBgpCmd(ctx context.Context, command string) (string, error)
FrrZebraCmd(ctx context.Context, command string, cmdTypeShow bool) (string, error)
FrrBgpCmd(ctx context.Context, command string, cmdTypeShow bool) (string, error)
Save(context.Context) error
Password(conn *telnet.Conn, delim string) error
EnterPrivileged(conn *telnet.Conn) error
Expand Down Expand Up @@ -115,16 +116,47 @@ func (n *FrrWrapper) ExitPrivileged(conn *telnet.Conn) error {
return conn.SkipUntil(">")
}

// checkFrrResult checks the vrf result
func checkFrrResult(data string, show bool) bool {
if show && data == "" {
return true
}
// // Trying to delete non exist VRF || Trying to delete non exist peer-group
if strings.Contains(data, "Can't find BGP instance") || strings.Contains(data, "Create the peer-group first") {
return false
}
patterns := []string{"error", "warning", "unknown", "ambiguous", "specified does not exist"}
lowerStr := strings.ToLower(data)
for _, pattern := range patterns {
if strings.Contains(lowerStr, pattern) {
return true
}
}
return false
}

// FrrZebraCmd connects to Zebra telnet with password and runs command
func (n *FrrWrapper) FrrZebraCmd(ctx context.Context, command string) (string, error) {
func (n *FrrWrapper) FrrZebraCmd(ctx context.Context, command string, cmdTypeShow bool) (string, error) {
// ports defined here https://docs.frrouting.org/en/latest/setup.html#services
return n.TelnetDialAndCommunicate(ctx, command, zebra)
cmdOutput, cmdError := n.TelnetDialAndCommunicate(ctx, command, zebra)
if cmdError != nil {
return cmdOutput, cmdError
} else if checkFrrResult(cmdOutput, cmdTypeShow) {
return cmdOutput, errors.New("error while running FrrZebraCmd")
}
return cmdOutput, cmdError
}

// FrrBgpCmd connects to Bgp telnet with password and runs command
func (n *FrrWrapper) FrrBgpCmd(ctx context.Context, command string) (string, error) {
func (n *FrrWrapper) FrrBgpCmd(ctx context.Context, command string, cmdTypeShow bool) (string, error) {
// ports defined here https://docs.frrouting.org/en/latest/setup.html#services
return n.TelnetDialAndCommunicate(ctx, command, bgpd)
cmdOutput, cmdError := n.TelnetDialAndCommunicate(ctx, command, bgpd)
if cmdError != nil {
return cmdOutput, cmdError
} else if checkFrrResult(cmdOutput, cmdTypeShow) {
return cmdOutput, errors.New("error while running FrrBgpCmd")
}
return cmdOutput, cmdError
}

// Save command save the current config to /etc/frr/frr.conf
Expand Down

0 comments on commit 12cd7c0

Please sign in to comment.