-
Notifications
You must be signed in to change notification settings - Fork 217
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat: Retina dataplane debug CLI #740
base: main
Are you sure you want to change the base?
Changes from 19 commits
15ff6f7
3874c48
003518c
06fbf67
6ee6480
25abb52
df47157
c4c37f8
a906f60
babd276
407dc24
f9a6e6e
747de43
0f77229
8712967
e6a2cdb
810481b
a682a75
23d8e57
a25e507
bfa5a83
ba9745f
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,13 @@ | ||
package bpf | ||
|
||
import "github.com/spf13/cobra" | ||
|
||
var Cmd = &cobra.Command{ | ||
Use: "bpf", | ||
Short: "BPF debug commands", | ||
} | ||
|
||
func init() { | ||
Cmd.AddCommand(featuresCmd) | ||
Cmd.AddCommand(qdiscCmd) | ||
} | ||
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
package bpf | ||
|
||
import "github.com/spf13/cobra" | ||
|
||
var Cmd = &cobra.Command{ | ||
Use: "bpf", | ||
Short: "BPF debug commands (Not supported on Windows)", | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,88 @@ | ||
package bpf | ||
|
||
import ( | ||
"fmt" | ||
|
||
"github.com/cilium/ebpf" | ||
"github.com/pkg/errors" | ||
) | ||
|
||
var ( | ||
eBPFMapList = []ebpf.MapType{ | ||
ebpf.Hash, | ||
ebpf.Array, | ||
ebpf.ProgramArray, | ||
ebpf.PerfEventArray, | ||
ebpf.PerCPUHash, | ||
ebpf.PerCPUArray, | ||
ebpf.StackTrace, | ||
ebpf.CGroupArray, | ||
ebpf.LRUHash, | ||
ebpf.LRUCPUHash, | ||
ebpf.LPMTrie, | ||
ebpf.ArrayOfMaps, | ||
ebpf.HashOfMaps, | ||
ebpf.DevMap, | ||
ebpf.SockMap, | ||
ebpf.CPUMap, | ||
ebpf.XSKMap, | ||
ebpf.SockHash, | ||
ebpf.CGroupStorage, | ||
ebpf.ReusePortSockArray, | ||
ebpf.PerCPUCGroupStorage, | ||
ebpf.Queue, | ||
ebpf.Stack, | ||
ebpf.SkStorage, | ||
ebpf.DevMapHash, | ||
ebpf.StructOpsMap, | ||
ebpf.RingBuf, | ||
ebpf.InodeStorage, | ||
ebpf.TaskStorage, | ||
} | ||
|
||
eBPFProgramList = []ebpf.ProgramType{ | ||
ebpf.SocketFilter, | ||
ebpf.Kprobe, | ||
ebpf.SchedCLS, | ||
ebpf.SchedACT, | ||
ebpf.TracePoint, | ||
ebpf.XDP, | ||
ebpf.PerfEvent, | ||
ebpf.CGroupSKB, | ||
ebpf.CGroupSock, | ||
ebpf.LWTIn, | ||
ebpf.LWTOut, | ||
ebpf.LWTXmit, | ||
ebpf.SockOps, | ||
ebpf.SkSKB, | ||
ebpf.CGroupDevice, | ||
ebpf.SkMsg, | ||
ebpf.RawTracepoint, | ||
ebpf.CGroupSockAddr, | ||
ebpf.LWTSeg6Local, | ||
ebpf.LircMode2, | ||
ebpf.SkReuseport, | ||
ebpf.FlowDissector, | ||
ebpf.CGroupSysctl, | ||
ebpf.RawTracepointWritable, | ||
ebpf.CGroupSockopt, | ||
ebpf.Tracing, | ||
ebpf.StructOps, | ||
ebpf.Extension, | ||
ebpf.LSM, | ||
ebpf.SkLookup, | ||
ebpf.Syscall, | ||
ebpf.Netfilter, | ||
} | ||
) | ||
|
||
func isSupported(err error) string { | ||
if errors.Is(err, ebpf.ErrNotSupported) { | ||
return "not supported" | ||
} | ||
return "supported" | ||
} | ||
|
||
func getLinuxKernelVersion(versionCode uint32) string { | ||
return fmt.Sprintf("%d.%d.%d", versionCode>>16, (versionCode>>8)&0xff, versionCode&0xff) //nolint:gomnd // bit shifting | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,61 @@ | ||
package bpf | ||
|
||
import ( | ||
"fmt" | ||
|
||
"github.com/cilium/ebpf" | ||
"github.com/cilium/ebpf/features" | ||
"github.com/pkg/errors" | ||
"github.com/spf13/cobra" | ||
) | ||
|
||
// featuresCmd outputs available BPF features on the host | ||
var featuresCmd = &cobra.Command{ | ||
Use: "features", | ||
Short: "Output available BPF features on the host", | ||
RunE: func(*cobra.Command, []string) error { | ||
linuxVersion, err := features.LinuxVersionCode() | ||
if err != nil { | ||
return errors.Wrap(err, "failed to get Linux version code") | ||
} | ||
fmt.Printf("Linux kernel version: %s\n", getLinuxKernelVersion(linuxVersion)) | ||
|
||
fmt.Println("--------------------------------------------------") | ||
|
||
err = features.HaveBoundedLoops() | ||
if err != nil && !errors.Is(err, ebpf.ErrNotSupported) { | ||
return errors.Wrap(err, "failed to check for bounded loops") | ||
} | ||
fmt.Printf("Bounded loops: %s\n", isSupported(err)) | ||
|
||
fmt.Println("--------------------------------------------------") | ||
|
||
err = features.HaveLargeInstructions() | ||
if err != nil && !errors.Is(err, ebpf.ErrNotSupported) { | ||
return errors.Wrap(err, "failed to check for large instructions") | ||
} | ||
fmt.Printf("Large instructions: %s\n", isSupported(err)) | ||
|
||
fmt.Println("--------------------------------------------------") | ||
fmt.Println("eBPF map availability:") | ||
for _, mt := range eBPFMapList { | ||
err = features.HaveMapType(mt) | ||
if err != nil && !errors.Is(err, ebpf.ErrNotSupported) { | ||
return errors.Wrapf(err, "failed to check for map type %s", mt.String()) | ||
} | ||
fmt.Printf("%s: %s\n", mt.String(), isSupported(err)) | ||
} | ||
|
||
fmt.Println("--------------------------------------------------") | ||
fmt.Println("eBPF program types availability:") | ||
for _, pt := range eBPFProgramList { | ||
err = features.HaveProgramType(pt) | ||
if err != nil && !errors.Is(err, ebpf.ErrNotSupported) { | ||
return errors.Wrapf(err, "failed to check for program type %s", pt.String()) | ||
} | ||
fmt.Printf("%s: %s\n", pt.String(), isSupported(err)) | ||
} | ||
Comment on lines
+40
to
+57
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. IMO, it's not important to list all map/prog types. We can just list all maps and programs, and that would be enough information. For specific maps (like conntrack), we may want to print entries, but listing by types is not that helpful. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. this is more so for users to check which maps and programs can be initialize on their host, but yes i do plan to have a command that will list out maps/programs running |
||
|
||
return nil | ||
}, | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,129 @@ | ||
package bpf | ||
|
||
import ( | ||
"fmt" | ||
"net" | ||
"os" | ||
|
||
tc "github.com/florianl/go-tc" | ||
"github.com/florianl/go-tc/core" | ||
"github.com/mdlayher/netlink" | ||
"github.com/pkg/errors" | ||
"github.com/spf13/cobra" | ||
"golang.org/x/sys/unix" | ||
) | ||
|
||
// val is a struct to hold qdiscs and filters for an interface | ||
type val struct { | ||
qdiscs []tc.Object | ||
egressfilters []tc.Object | ||
ingressfilters []tc.Object | ||
} | ||
|
||
var ( | ||
ifaceName string | ||
ifaceToQdiscsAndFiltersMap = make(map[string]*val) | ||
qdiscCmd = &cobra.Command{ | ||
Use: "tc", | ||
Short: "Output all qdiscs and attached bpf programs on each interface on the host", | ||
RunE: func(*cobra.Command, []string) error { | ||
// open a rtnetlink socket | ||
rtnl, err := tc.Open(&tc.Config{}) | ||
if err != nil { | ||
return errors.Wrap(err, "could not open rtnetlink socket") | ||
} | ||
defer func() { | ||
if err = rtnl.Close(); err != nil { | ||
fmt.Fprintf(os.Stderr, "could not close rtnetlink socket: %v\n", err) | ||
} | ||
}() | ||
|
||
err = rtnl.SetOption(netlink.ExtendedAcknowledge, true) | ||
if err != nil { | ||
return errors.Wrap(err, "could not set NETLINK_EXT_ACK option") | ||
} | ||
|
||
qdiscs, err := rtnl.Qdisc().Get() | ||
if err != nil { | ||
return errors.Wrap(err, "could not get qdiscs") | ||
} | ||
|
||
for _, qdisc := range qdiscs { | ||
iface, err := net.InterfaceByIndex(int(qdisc.Ifindex)) | ||
if err != nil { | ||
return errors.Wrap(err, "could not get interface by index") | ||
} | ||
if _, ok := ifaceToQdiscsAndFiltersMap[iface.Name]; !ok { | ||
ifaceToQdiscsAndFiltersMap[iface.Name] = &val{} | ||
} | ||
ifaceToQdiscsAndFiltersMap[iface.Name].qdiscs = append(ifaceToQdiscsAndFiltersMap[iface.Name].qdiscs, qdisc) | ||
|
||
ingressFilters, err := rtnl.Filter().Get(&tc.Msg{ | ||
Family: unix.AF_UNSPEC, | ||
Ifindex: uint32(iface.Index), | ||
Handle: 0, | ||
Parent: core.BuildHandle(tc.HandleRoot, tc.HandleMinIngress), | ||
Info: 0x10300, // nolint:gomnd // info | ||
}) | ||
if err != nil { | ||
return errors.Wrap(err, "could not get ingress filters for interface") | ||
} | ||
ifaceToQdiscsAndFiltersMap[iface.Name].ingressfilters = append(ifaceToQdiscsAndFiltersMap[iface.Name].ingressfilters, ingressFilters...) | ||
|
||
egressFilters, err := rtnl.Filter().Get(&tc.Msg{ | ||
Family: unix.AF_UNSPEC, | ||
Ifindex: uint32(iface.Index), | ||
Handle: 1, | ||
Parent: core.BuildHandle(tc.HandleRoot, tc.HandleMinEgress), | ||
Info: 0x10300, // nolint:gomnd // info | ||
}) | ||
if err != nil { | ||
return errors.Wrap(err, "could not get egress filters for interface") | ||
} | ||
ifaceToQdiscsAndFiltersMap[iface.Name].egressfilters = append(ifaceToQdiscsAndFiltersMap[iface.Name].egressfilters, egressFilters...) | ||
} | ||
|
||
if ifaceName != "" { | ||
if val, ok := ifaceToQdiscsAndFiltersMap[ifaceName]; ok { | ||
fmt.Printf("Interface: %s\n", ifaceName) | ||
fmt.Printf("Qdiscs:\n") | ||
for _, qdisc := range val.qdiscs { | ||
fmt.Printf(" %s\n", qdisc.Kind) | ||
} | ||
fmt.Printf("Ingress filters:\n") | ||
for _, ingressFilter := range val.ingressfilters { | ||
fmt.Printf(" %+v\n", ingressFilter) | ||
} | ||
fmt.Printf("Egress filters:\n") | ||
for _, egressFilter := range val.egressfilters { | ||
fmt.Printf(" %+v\n", egressFilter) | ||
} | ||
} else { | ||
fmt.Printf("Interface %s not found\n", ifaceName) | ||
} | ||
} else { | ||
for iface, val := range ifaceToQdiscsAndFiltersMap { | ||
fmt.Printf("Interface: %s\n", iface) | ||
fmt.Printf("Qdiscs:\n") | ||
for _, qdisc := range val.qdiscs { | ||
fmt.Printf(" %s\n", qdisc.Kind) | ||
} | ||
fmt.Printf("Ingress filters:\n") | ||
for _, ingressFilter := range val.ingressfilters { | ||
fmt.Printf(" %+v\n", ingressFilter) | ||
} | ||
fmt.Printf("Egress filters:\n") | ||
for _, egressFilter := range val.egressfilters { | ||
fmt.Printf(" %+v\n", egressFilter) | ||
} | ||
} | ||
} | ||
|
||
return nil | ||
}, | ||
} | ||
) | ||
|
||
func init() { | ||
qdiscCmd.Flags().StringVarP(&ifaceName, "interface", "i", "", "Filter output to a specific interface") | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,13 @@ | ||
package conntrack | ||
|
||
import "github.com/spf13/cobra" | ||
|
||
var Cmd = &cobra.Command{ | ||
Use: "conntrack", | ||
Short: "Conntrack debug commands", | ||
} | ||
|
||
func init() { | ||
Cmd.AddCommand(dump) | ||
Cmd.AddCommand(stats) | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
package conntrack | ||
|
||
import "github.com/spf13/cobra" | ||
|
||
var Cmd = &cobra.Command{ | ||
Use: "conntrack", | ||
Short: "Conntrack debug commands (Not supported on Windows)", | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,14 @@ | ||
package conntrack | ||
|
||
import ( | ||
"github.com/microsoft/retina/pkg/plugin/conntrack" | ||
"github.com/spf13/cobra" | ||
) | ||
|
||
var dump = &cobra.Command{ | ||
Use: "dump", | ||
Short: "Dump all conntrack entries", | ||
RunE: func(*cobra.Command, []string) error { | ||
return conntrack.Dump() | ||
}, | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,14 @@ | ||
package conntrack | ||
|
||
import ( | ||
"github.com/microsoft/retina/pkg/plugin/conntrack" | ||
"github.com/spf13/cobra" | ||
) | ||
|
||
var stats = &cobra.Command{ | ||
Use: "stats", | ||
Short: "Print conntrack stats", | ||
RunE: func(*cobra.Command, []string) error { | ||
return conntrack.Stats() | ||
}, | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,17 @@ | ||
package debug | ||
|
||
import ( | ||
"github.com/microsoft/retina/cli/cmd/debug/bpf" | ||
"github.com/microsoft/retina/cli/cmd/debug/conntrack" | ||
"github.com/spf13/cobra" | ||
) | ||
|
||
var Cmd = &cobra.Command{ | ||
Use: "debug", | ||
Short: "Dataplane debug commands", | ||
} | ||
|
||
func init() { | ||
Cmd.AddCommand(conntrack.Cmd) | ||
Cmd.AddCommand(bpf.Cmd) | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Let's add a
status
command, showcasing the important one-liners:There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
you can already get flow rate by exec
hubble status
on the agent so it might be overkill?