vpp/extras/vpp_stats_fs/stats_fs.go
Arthur de Kerhor 9cfbd3b786 misc: bug fixes and improvements for stats Fuse fs
Added syslogs
Added support for symlinks
Relocated make commands in a local Makefile
Dumping stats on index instead of paths
Updated README
Added go.mod and go.sum with relevant dependencies for the module

Type: fix
Change-Id: I2c91317939b2f4d765771ab7038372ae27d3109d
Signed-off-by: Arthur de Kerhor <arthurdekerhor@gmail.com>
2021-06-03 07:25:04 +00:00

227 lines
6.7 KiB
Go

/*
* Copyright (c) 2021 Cisco Systems and/or its affiliates.
* 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.
*/
/*Go-FUSE allows us to define the behaviour of our filesystem by recoding any primitive function we need.
*The structure of the filesystem is constructed as a tree.
*Each type of nodes (root, directory, file) follows its own prmitives.
*/
package main
import (
"context"
"fmt"
"path"
"path/filepath"
"strings"
"syscall"
"time"
"github.com/hanwen/go-fuse/v2/fs"
"github.com/hanwen/go-fuse/v2/fuse"
"git.fd.io/govpp.git/adapter"
"git.fd.io/govpp.git/adapter/statsclient"
)
func updateDir(ctx context.Context, n *fs.Inode, cl *statsclient.StatsClient, dirPath string) syscall.Errno {
stats, err := cl.PrepareDir(dirPath)
if err != nil {
LogMsg(fmt.Sprintf("Listing stats index failed: %v\n", err))
return syscall.EAGAIN
}
n.Operations().(*dirNode).epoch = stats.Epoch
n.RmAllChildren()
for _, entry := range stats.Entries {
localPath := strings.TrimPrefix(string(entry.Name), dirPath)
dirPath, base := filepath.Split(localPath)
parent := n
for _, component := range strings.Split(dirPath, "/") {
if len(component) == 0 {
continue
}
child := parent.GetChild(component)
if child == nil {
child = parent.NewInode(ctx, &dirNode{client: cl, epoch: stats.Epoch},
fs.StableAttr{Mode: fuse.S_IFDIR})
parent.AddChild(component, child, true)
} else {
child.Operations().(*dirNode).epoch = stats.Epoch
}
parent = child
}
filename := strings.Replace(base, " ", "_", -1)
child := parent.GetChild(filename)
if child == nil {
child := parent.NewPersistentInode(ctx, &statNode{client: cl, index: entry.Index}, fs.StableAttr{})
parent.AddChild(filename, child, true)
}
}
return 0
}
func getCounterContent(index uint32, client *statsclient.StatsClient) (content string, status syscall.Errno) {
content = ""
statsDir, err := client.PrepareDirOnIndex(index)
if err != nil {
LogMsg(fmt.Sprintf("Dumping stats on index failed: %v\n", err))
return content, syscall.EAGAIN
}
if len(statsDir.Entries) != 1 {
return content, syscall.ENOENT
}
result := statsDir.Entries[0]
if result.Data == nil {
return content, 0
}
switch result.Type {
case adapter.ScalarIndex:
stats := result.Data.(adapter.ScalarStat)
content = fmt.Sprintf("%.2f\n", stats)
case adapter.ErrorIndex:
stats := result.Data.(adapter.ErrorStat)
content = fmt.Sprintf("%-16s%s\n", "Index", "Count")
for i, value := range stats {
content += fmt.Sprintf("%-16d%d\n", i, value)
}
case adapter.SimpleCounterVector:
stats := result.Data.(adapter.SimpleCounterStat)
content = fmt.Sprintf("%-16s%-16s%s\n", "Thread", "Index", "Packets")
for i, vector := range stats {
for j, value := range vector {
content += fmt.Sprintf("%-16d%-16d%d\n", i, j, value)
}
}
case adapter.CombinedCounterVector:
stats := result.Data.(adapter.CombinedCounterStat)
content = fmt.Sprintf("%-16s%-16s%-16s%s\n", "Thread", "Index", "Packets", "Bytes")
for i, vector := range stats {
for j, value := range vector {
content += fmt.Sprintf("%-16d%-16d%-16d%d\n", i, j, value[0], value[1])
}
}
case adapter.NameVector:
stats := result.Data.(adapter.NameStat)
content = fmt.Sprintf("%-16s%s\n", "Index", "Name")
for i, value := range stats {
content += fmt.Sprintf("%-16d%s\n", i, string(value))
}
default:
content = fmt.Sprintf("Unknown stat type: %d\n", result.Type)
//For now, the empty type (file deleted) is not implemented in GoVPP
return content, syscall.ENOENT
}
return content, fs.OK
}
//The dirNode structure represents directories
type dirNode struct {
fs.Inode
client *statsclient.StatsClient
epoch int64
}
var _ = (fs.NodeOpendirer)((*dirNode)(nil))
var _ = (fs.NodeGetattrer)((*dirNode)(nil))
var _ = (fs.NodeOnAdder)((*dirNode)(nil))
func (dn *dirNode) OnAdd(ctx context.Context) {
if dn.Inode.IsRoot() {
updateDir(ctx, &dn.Inode, dn.client, "/")
}
}
func (dn *dirNode) Getattr(ctx context.Context, f fs.FileHandle, out *fuse.AttrOut) syscall.Errno {
out.Mtime = uint64(time.Now().Unix())
out.Atime = out.Mtime
out.Ctime = out.Mtime
return 0
}
func (dn *dirNode) Opendir(ctx context.Context) syscall.Errno {
var status syscall.Errno = syscall.F_OK
var sleepTime time.Duration = 10 * time.Millisecond
newEpoch, inProgress := dn.client.GetEpoch()
for inProgress {
newEpoch, inProgress = dn.client.GetEpoch()
time.Sleep(sleepTime)
sleepTime = sleepTime * 2
}
//We check that the directory epoch is up to date
if dn.epoch != newEpoch {
//directoryPath is the path to the current directory from root
directoryPath := path.Clean("/" + dn.Inode.Path(nil) + "/")
status = updateDir(ctx, &dn.Inode, dn.client, directoryPath)
}
return status
}
//The statNode structure represents counters
type statNode struct {
fs.Inode
client *statsclient.StatsClient
index uint32
}
var _ = (fs.NodeOpener)((*statNode)(nil))
var _ = (fs.NodeGetattrer)((*statNode)(nil))
func (fh *statNode) Getattr(ctx context.Context, f fs.FileHandle, out *fuse.AttrOut) syscall.Errno {
out.Mtime = uint64(time.Now().Unix())
out.Atime = out.Mtime
out.Ctime = out.Mtime
return 0
}
//When a file is opened, the correpsonding counter value is dumped and a file handle is created
func (sn *statNode) Open(ctx context.Context, flags uint32) (fs.FileHandle, uint32, syscall.Errno) {
content, status := getCounterContent(sn.index, sn.client)
if status == syscall.ENOENT {
_, parent := sn.Inode.Parent()
parent.RmChild(sn.Inode.Path(parent))
}
return &statFH{data: []byte(content)}, fuse.FOPEN_DIRECT_IO, status
}
/* The statFH structure aims at dislaying the counters dynamically.
* It allows the Kernel to read data as I/O without having to specify files sizes, as they may evolve dynamically.
*/
type statFH struct {
data []byte
}
var _ = (fs.FileReader)((*statFH)(nil))
func (fh *statFH) Read(ctx context.Context, data []byte, off int64) (fuse.ReadResult, syscall.Errno) {
end := int(off) + len(data)
if end > len(fh.data) {
end = len(fh.data)
}
return fuse.ReadResultData(fh.data[off:end]), fs.OK
}
//NewStatsFileSystem creates the fs for the stat segment.
func NewStatsFileSystem(sc *statsclient.StatsClient) (root fs.InodeEmbedder, err error) {
return &dirNode{client: sc}, nil
}