Initial stab at snapshot inspect key breakdown

This commit is contained in:
Joel Watson 2020-11-03 18:00:44 -06:00
parent ebe0ffce30
commit bc1a55cd09
2 changed files with 87 additions and 11 deletions

View File

@ -15,7 +15,7 @@ const (
) )
type Formatter interface { type Formatter interface {
Format(*OutputFormat) (string, error) Format(*OutputFormat, *OutputFormat, bool) (string, error)
} }
func GetSupportedFormats() []string { func GetSupportedFormats() []string {
@ -38,7 +38,7 @@ func NewFormatter(format string) (Formatter, error) {
} }
} }
func (_ *prettyFormatter) Format(info *OutputFormat) (string, error) { func (_ *prettyFormatter) Format(info *OutputFormat, kvInfo *OutputFormat, detailed bool) (string, error) {
var b bytes.Buffer var b bytes.Buffer
tw := tabwriter.NewWriter(&b, 8, 8, 6, ' ', 0) tw := tabwriter.NewWriter(&b, 8, 8, 6, ' ', 0)
@ -60,6 +60,25 @@ func (_ *prettyFormatter) Format(info *OutputFormat) (string, error) {
if err := tw.Flush(); err != nil { if err := tw.Flush(); err != nil {
return b.String(), err return b.String(), err
} }
if detailed {
kvtw := tabwriter.NewWriter(&b, 30, 8, 12, ' ', 0)
fmt.Fprintf(kvtw, "\n")
fmt.Fprintln(kvtw, "\n Key Name\tCount\tSize\t")
fmt.Fprintf(kvtw, " %s\t%s\t%s\t", "----", "----", "----")
// For each different type generate new output
for _, s := range kvInfo.Stats {
fmt.Fprintf(kvtw, "\n %s\t%d\t%s\t", s.Name, s.Count, ByteSize(uint64(s.Sum)))
}
fmt.Fprintf(kvtw, "\n %s\t%s\t%s\t", "----", "----", "----")
fmt.Fprintf(kvtw, "\n Total\t\t%s\t", ByteSize(uint64(kvInfo.TotalSize)))
if err := kvtw.Flush(); err != nil {
return b.String(), err
}
}
return b.String(), nil return b.String(), nil
} }
@ -69,7 +88,7 @@ func newJSONFormatter() Formatter {
return &jsonFormatter{} return &jsonFormatter{}
} }
func (_ *jsonFormatter) Format(info *OutputFormat) (string, error) { func (_ *jsonFormatter) Format(info *OutputFormat, infoKV *OutputFormat, detailed bool) (string, error) {
b, err := json.MarshalIndent(info, "", " ") b, err := json.MarshalIndent(info, "", " ")
if err != nil { if err != nil {
return "", fmt.Errorf("Failed to marshal original snapshot stats: %v", err) return "", fmt.Errorf("Failed to marshal original snapshot stats: %v", err)

View File

@ -29,10 +29,18 @@ type cmd struct {
flags *flag.FlagSet flags *flag.FlagSet
help string help string
format string format string
// flags
detailed bool
kvDepth int
} }
func (c *cmd) init() { func (c *cmd) init() {
c.flags = flag.NewFlagSet("", flag.ContinueOnError) c.flags = flag.NewFlagSet("", flag.ContinueOnError)
c.flags.BoolVar(&c.detailed, "detailed", false,
"Provides detailed information about KV store data.")
c.flags.IntVar(&c.kvDepth, "kv-depth", 2,
"The key prefix depth used to breakdown KV store data. Defaults to 2.")
c.flags.StringVar( c.flags.StringVar(
&c.format, &c.format,
"format", "format",
@ -57,6 +65,7 @@ type MetadataInfo struct {
type OutputFormat struct { type OutputFormat struct {
Meta *MetadataInfo Meta *MetadataInfo
Stats []typeStats Stats []typeStats
KStats []typeStats
TotalSize int TotalSize int
} }
@ -101,7 +110,7 @@ func (c *cmd) Run(args []string) int {
} }
}() }()
stats, totalSize, err := enhance(readFile) stats, kstats, totalSize, err := enhance(readFile, c.detailed, c.kvDepth)
if err != nil { if err != nil {
c.UI.Error(fmt.Sprintf("Error extracting snapshot data: %s", err)) c.UI.Error(fmt.Sprintf("Error extracting snapshot data: %s", err))
return 1 return 1
@ -122,14 +131,20 @@ func (c *cmd) Run(args []string) int {
} }
//Restructures stats given above to be human readable //Restructures stats given above to be human readable
formattedStats := generatetypeStats(stats) formattedStats, formattedKStats := generatetypeStats(stats, kstats, c.detailed)
in := &OutputFormat{ in := &OutputFormat{
Meta: metaformat, Meta: metaformat,
Stats: formattedStats, Stats: formattedStats,
TotalSize: totalSize, TotalSize: totalSize,
} }
out, err := formatter.Format(in) inKV := &OutputFormat{
Meta: metaformat,
Stats: formattedKStats,
TotalSize: totalSize,
}
out, err := formatter.Format(in, inKV, c.detailed)
if err != nil { if err != nil {
c.UI.Error(err.Error()) c.UI.Error(err.Error())
return 1 return 1
@ -145,7 +160,7 @@ type typeStats struct {
Count int Count int
} }
func generatetypeStats(info map[structs.MessageType]typeStats) []typeStats { func generatetypeStats(info map[structs.MessageType]typeStats, kvInfo map[string]typeStats, detailed bool) ([]typeStats, []typeStats) {
ss := make([]typeStats, 0, len(info)) ss := make([]typeStats, 0, len(info))
for _, s := range info { for _, s := range info {
@ -155,7 +170,20 @@ func generatetypeStats(info map[structs.MessageType]typeStats) []typeStats {
// Sort the stat slice // Sort the stat slice
sort.Slice(ss, func(i, j int) bool { return ss[i].Sum > ss[j].Sum }) sort.Slice(ss, func(i, j int) bool { return ss[i].Sum > ss[j].Sum })
return ss if detailed {
ks := make([]typeStats, 0, len(kvInfo))
for _, s := range kvInfo {
ks = append(ks, s)
}
// Sort the kv stat slice
sort.Slice(ks, func(i, j int) bool { return ks[i].Sum > ks[j].Sum })
return ss, ks
}
return ss, nil
} }
// countingReader helps keep track of the bytes we have read // countingReader helps keep track of the bytes we have read
@ -175,8 +203,9 @@ func (r *countingReader) Read(p []byte) (n int, err error) {
// enhance utilizes ReadSnapshot to populate the struct with // enhance utilizes ReadSnapshot to populate the struct with
// all of the snapshot's itemized data // all of the snapshot's itemized data
func enhance(file io.Reader) (map[structs.MessageType]typeStats, int, error) { func enhance(file io.Reader, detailed bool, kvDepth int) (map[structs.MessageType]typeStats, map[string]typeStats, int, error) {
stats := make(map[structs.MessageType]typeStats) stats := make(map[structs.MessageType]typeStats)
kstats := make(map[string]typeStats)
cr := &countingReader{wrappedReader: file} cr := &countingReader{wrappedReader: file}
totalSize := 0 totalSize := 0
handler := func(header *fsm.SnapshotHeader, msg structs.MessageType, dec *codec.Decoder) error { handler := func(header *fsm.SnapshotHeader, msg structs.MessageType, dec *codec.Decoder) error {
@ -185,6 +214,7 @@ func enhance(file io.Reader) (map[structs.MessageType]typeStats, int, error) {
if s.Name == "" { if s.Name == "" {
s.Name = name s.Name = name
} }
var val interface{} var val interface{}
err := dec.Decode(&val) err := dec.Decode(&val)
if err != nil { if err != nil {
@ -196,12 +226,39 @@ func enhance(file io.Reader) (map[structs.MessageType]typeStats, int, error) {
s.Count++ s.Count++
totalSize = cr.read totalSize = cr.read
stats[msg] = s stats[msg] = s
if detailed {
if s.Name == "KVS" {
switch val := val.(type) {
case map[string]interface{}:
fmt.Println("map-match")
for k, v := range val {
depth := kvDepth
if k == "Key" {
split := strings.Split(v.(string), "/")
if depth > len(split) {
depth = len(split)
}
prefix := strings.Join(split[0:depth], "/")
kvs := kstats[prefix]
if kvs.Name == "" {
kvs.Name = prefix
}
kvs.Sum += size
kvs.Count++
kstats[prefix] = kvs
}
}
}
}
}
return nil return nil
} }
if err := fsm.ReadSnapshot(cr, handler); err != nil { if err := fsm.ReadSnapshot(cr, handler); err != nil {
return nil, 0, err return nil, nil, 0, err
} }
return stats, totalSize, nil return stats, kstats, totalSize, nil
} }