2018-11-21 18:46:54 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
|
|
|
"git.fd.io/govpp.git"
|
|
|
|
"git.fd.io/govpp.git/adapter"
|
|
|
|
"git.fd.io/govpp.git/adapter/vppapiclient"
|
|
|
|
"git.fd.io/govpp.git/api"
|
|
|
|
"git.fd.io/govpp.git/core"
|
|
|
|
"git.fd.io/govpp.git/examples/bin_api/interfaces"
|
|
|
|
"git.fd.io/govpp.git/examples/bin_api/vpe"
|
|
|
|
"log"
|
|
|
|
)
|
|
|
|
|
|
|
|
//////////////////////////////////////
|
|
|
|
///////// Data structs ///////////
|
|
|
|
//////////////////////////////////////
|
|
|
|
|
|
|
|
const defaultStatsSocketPath = "/run/vpp/stats.sock"
|
|
|
|
const defaultShmPrefix = ""
|
|
|
|
|
|
|
|
func parseMacAddress(l2Address []byte, l2AddressLength uint32) string {
|
|
|
|
var mac string
|
|
|
|
for i := uint32(0); i < l2AddressLength; i++ {
|
|
|
|
mac += fmt.Sprintf("%02x", l2Address[i])
|
|
|
|
if i < l2AddressLength-1 {
|
|
|
|
mac += ":"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return mac
|
|
|
|
}
|
|
|
|
|
|
|
|
type interfaceStats struct {
|
|
|
|
TxBytes uint64
|
|
|
|
TxPackets uint64
|
|
|
|
TxErrors uint64
|
|
|
|
RxBytes uint64
|
|
|
|
RxPackets uint64
|
|
|
|
RxErrors uint64
|
|
|
|
Drops uint64
|
|
|
|
Punts uint64
|
|
|
|
}
|
|
|
|
|
|
|
|
type vppInterface struct {
|
|
|
|
interfaces.SwInterfaceDetails
|
|
|
|
Stats interfaceStats
|
|
|
|
}
|
|
|
|
|
|
|
|
type vppConnector struct {
|
|
|
|
statsSocketPath string
|
|
|
|
shmPrefix string
|
|
|
|
|
|
|
|
conn *core.Connection
|
|
|
|
api api.Channel
|
|
|
|
stats adapter.StatsAPI
|
|
|
|
|
|
|
|
VppDetails vpe.ShowVersionReply
|
|
|
|
Interfaces []*vppInterface
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////
|
|
|
|
///////// VPP workflow ///////////
|
|
|
|
//////////////////////////////////////
|
|
|
|
|
|
|
|
func (v *vppConnector) getVppVersion() error {
|
|
|
|
if err := v.api.SendRequest(&vpe.ShowVersion{}).ReceiveReply(&v.VppDetails); err != nil {
|
|
|
|
return fmt.Errorf("failed to fetch vpp version: %v", err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (v *vppConnector) getInterfaces() error {
|
|
|
|
ifCtx := v.api.SendMultiRequest(&interfaces.SwInterfaceDump{})
|
|
|
|
for {
|
|
|
|
ifDetails := interfaces.SwInterfaceDetails{}
|
|
|
|
stop, err := ifCtx.ReceiveReply(&ifDetails)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("failed to fetch vpp interface: %v", err)
|
|
|
|
}
|
|
|
|
if stop {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
v.Interfaces = append(v.Interfaces, &vppInterface{SwInterfaceDetails: ifDetails})
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (v *vppConnector) connect() (err error) {
|
|
|
|
if v.conn, err = govpp.Connect(v.shmPrefix); err != nil {
|
|
|
|
return fmt.Errorf("failed to connect to vpp: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if v.api, err = v.conn.NewAPIChannel(); err != nil {
|
|
|
|
return fmt.Errorf("failed to create api channel: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
v.stats = vppapiclient.NewStatClient(v.statsSocketPath)
|
|
|
|
if err = v.stats.Connect(); err != nil {
|
|
|
|
return fmt.Errorf("failed to connect to Stats adapter: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (v *vppConnector) disconnect() {
|
|
|
|
if v.stats != nil {
|
|
|
|
v.stats.Disconnect()
|
|
|
|
}
|
|
|
|
if v.conn != nil {
|
|
|
|
v.conn.Disconnect()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (v *vppConnector) reduceCombinedCounters(stat *adapter.StatEntry) *[]adapter.CombinedCounter {
|
|
|
|
counters := stat.Data.(adapter.CombinedCounterStat)
|
|
|
|
stats := make([]adapter.CombinedCounter, len(v.Interfaces))
|
|
|
|
for _, workerStats := range counters {
|
2018-12-06 10:51:18 +02:00
|
|
|
for i := 0; i < len(v.Interfaces); i++ {
|
|
|
|
stats[i].Bytes += workerStats[i].Bytes
|
|
|
|
stats[i].Packets += workerStats[i].Packets
|
2018-11-21 18:46:54 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return &stats
|
|
|
|
}
|
|
|
|
|
|
|
|
func (v *vppConnector) reduceSimpleCounters(stat *adapter.StatEntry) *[]adapter.Counter {
|
|
|
|
counters := stat.Data.(adapter.SimpleCounterStat)
|
|
|
|
stats := make([]adapter.Counter, len(v.Interfaces))
|
|
|
|
for _, workerStats := range counters {
|
2018-12-06 10:51:18 +02:00
|
|
|
for i := 0; i < len(v.Interfaces); i++ {
|
|
|
|
stats[i] += workerStats[i]
|
2018-11-21 18:46:54 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return &stats
|
|
|
|
}
|
|
|
|
|
|
|
|
func (v *vppConnector) getStatsForAllInterfaces() error {
|
|
|
|
statsDump, err := v.stats.DumpStats("/if")
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("failed to dump vpp Stats: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
stats := func(i int) *interfaceStats { return &v.Interfaces[uint32(i)].Stats }
|
|
|
|
|
|
|
|
for _, stat := range statsDump {
|
|
|
|
switch stat.Name {
|
|
|
|
case "/if/tx":
|
|
|
|
{
|
|
|
|
for i, counter := range *v.reduceCombinedCounters(stat) {
|
|
|
|
stats(i).TxBytes = uint64(counter.Bytes)
|
|
|
|
stats(i).TxPackets = uint64(counter.Packets)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case "/if/rx":
|
|
|
|
{
|
|
|
|
for i, counter := range *v.reduceCombinedCounters(stat) {
|
|
|
|
stats(i).RxBytes = uint64(counter.Bytes)
|
|
|
|
stats(i).RxPackets = uint64(counter.Packets)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case "/if/tx-error":
|
|
|
|
{
|
|
|
|
for i, counter := range *v.reduceSimpleCounters(stat) {
|
|
|
|
stats(i).TxErrors = uint64(counter)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case "/if/rx-error":
|
|
|
|
{
|
|
|
|
for i, counter := range *v.reduceSimpleCounters(stat) {
|
|
|
|
stats(i).RxErrors = uint64(counter)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case "/if/drops":
|
|
|
|
{
|
|
|
|
for i, counter := range *v.reduceSimpleCounters(stat) {
|
|
|
|
stats(i).Drops = uint64(counter)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case "/if/punt":
|
|
|
|
{
|
|
|
|
for i, counter := range *v.reduceSimpleCounters(stat) {
|
|
|
|
stats(i).Punts = uint64(counter)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
statsSocketPathPtr := flag.String("stats_socket_path", defaultStatsSocketPath, "Path to vpp stats socket")
|
|
|
|
shmPrefixPtr := flag.String("shm_prefix", defaultShmPrefix, "Shared memory prefix (advanced)")
|
|
|
|
flag.Parse()
|
|
|
|
|
|
|
|
vppConn := &vppConnector{statsSocketPath: *statsSocketPathPtr, shmPrefix: *shmPrefixPtr}
|
|
|
|
defer vppConn.disconnect()
|
|
|
|
|
|
|
|
if err := vppConn.connect(); err != nil {
|
|
|
|
log.Fatalln(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := vppConn.getVppVersion(); err != nil {
|
|
|
|
log.Fatalln(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := vppConn.getInterfaces(); err != nil {
|
|
|
|
log.Fatalln(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := vppConn.getStatsForAllInterfaces(); err != nil {
|
|
|
|
log.Fatalln(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
jsonString, err := dumpToJSONString(vppConn)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalln(err)
|
|
|
|
}
|
|
|
|
fmt.Println(jsonString)
|
|
|
|
}
|