Initial stab at snapshot inspect key breakdown
This commit is contained in:
parent
ebe0ffce30
commit
bc1a55cd09
|
@ -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)
|
||||||
|
|
|
@ -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
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue