- package main
-
- import (
- "fmt"
- "io"
- "os"
- "text/tabwriter"
- "time"
-
- monitor "github.com/tendermint/tendermint/tools/tm-monitor/monitor"
- )
-
- const (
- // Default refresh rate - 200ms
- defaultRefreshRate = time.Millisecond * 200
- )
-
- // Ton - table of nodes.
- //
- // It produces the unordered list of nodes and updates it periodically.
- //
- // Default output is stdout, but it could be changed. Note if you want for
- // refresh to work properly, output must support [ANSI escape
- // codes](http://en.wikipedia.org/wiki/ANSI_escape_code).
- //
- // Ton was inspired by [Linux top
- // program](https://en.wikipedia.org/wiki/Top_(software)) as the name suggests.
- type Ton struct {
- monitor *monitor.Monitor
-
- RefreshRate time.Duration
- Output io.Writer
- quit chan struct{}
- }
-
- func NewTon(m *monitor.Monitor) *Ton {
- return &Ton{
- RefreshRate: defaultRefreshRate,
- Output: os.Stdout,
- quit: make(chan struct{}),
- monitor: m,
- }
- }
-
- func (o *Ton) Start() {
- clearScreen(o.Output)
- o.Print()
- go o.refresher()
- }
-
- func (o *Ton) Print() {
- moveCursor(o.Output, 1, 1)
- o.printHeader()
- fmt.Println()
- o.printTable()
- }
-
- func (o *Ton) Stop() {
- close(o.quit)
- }
-
- func (o *Ton) printHeader() {
- n := o.monitor.Network
- fmt.Fprintf(o.Output, "%v up %.2f%%\n", n.StartTime().Format(time.RFC1123Z), n.Uptime())
- fmt.Println()
- fmt.Fprintf(o.Output, "Height: %d\n", n.Height)
- fmt.Fprintf(o.Output, "Avg block time: %.3f ms\n", n.AvgBlockTime)
- fmt.Fprintf(o.Output, "Avg tx throughput: %.3f per sec\n", n.AvgTxThroughput)
- fmt.Fprintf(o.Output, "Avg block latency: %.3f ms\n", n.AvgBlockLatency)
- fmt.Fprintf(o.Output, "Active nodes: %d/%d (health: %s) Validators: %d\n", n.NumNodesMonitoredOnline, n.NumNodesMonitored, n.GetHealthString(), n.NumValidators)
- }
-
- func (o *Ton) printTable() {
- w := tabwriter.NewWriter(o.Output, 0, 0, 5, ' ', 0)
- fmt.Fprintln(w, "NAME\tHEIGHT\tBLOCK LATENCY\tONLINE\tVALIDATOR\t")
- for _, n := range o.monitor.Nodes {
- fmt.Fprintln(w, fmt.Sprintf("%s\t%d\t%.3f ms\t%v\t%v\t", n.Name, n.Height, n.BlockLatency, n.Online, n.IsValidator))
- }
- w.Flush()
- }
-
- // Internal loop for refreshing
- func (o *Ton) refresher() {
- for {
- select {
- case <-o.quit:
- return
- case <-time.After(o.RefreshRate):
- o.Print()
- }
- }
- }
-
- func clearScreen(w io.Writer) {
- fmt.Fprint(w, "\033[2J")
- }
-
- func moveCursor(w io.Writer, x int, y int) {
- fmt.Fprintf(w, "\033[%d;%dH", x, y)
- }
|