2020-03-01 23:09:00 +01:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
2020-03-08 21:43:01 +01:00
|
|
|
"log"
|
2020-03-08 20:58:13 +01:00
|
|
|
"os"
|
2020-03-01 23:09:00 +01:00
|
|
|
"time"
|
2020-03-05 14:32:50 +01:00
|
|
|
|
|
|
|
"git.lattuga.net/blallo/ruspa/tree"
|
2020-03-08 21:43:01 +01:00
|
|
|
"golang.org/x/crypto/ssh/terminal"
|
2020-03-01 23:09:00 +01:00
|
|
|
)
|
|
|
|
|
2020-03-03 14:22:12 +01:00
|
|
|
type UnitValue struct {
|
|
|
|
unit string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (u *UnitValue) String() string {
|
|
|
|
return u.unit
|
|
|
|
}
|
|
|
|
|
|
|
|
func (u *UnitValue) Set(value string) error {
|
|
|
|
switch value {
|
|
|
|
case "B":
|
|
|
|
u.unit = "B"
|
|
|
|
return nil
|
|
|
|
case "KB":
|
|
|
|
u.unit = "KB"
|
|
|
|
return nil
|
|
|
|
case "MB":
|
|
|
|
u.unit = "MB"
|
|
|
|
return nil
|
|
|
|
case "GB":
|
|
|
|
u.unit = "GB"
|
|
|
|
return nil
|
|
|
|
case "TB":
|
|
|
|
u.unit = "TB"
|
|
|
|
return nil
|
|
|
|
case "PB":
|
|
|
|
u.unit = "PB"
|
|
|
|
return nil
|
|
|
|
default:
|
2020-03-05 14:32:50 +01:00
|
|
|
return tree.ErrUnknownUnit
|
2020-03-03 14:22:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-19 00:11:11 +01:00
|
|
|
type LimitValue struct {
|
|
|
|
limit int64
|
|
|
|
input string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *LimitValue) String() string {
|
|
|
|
return fmt.Sprintf("%d", l.limit)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *LimitValue) Set(size string) error {
|
|
|
|
value, err := tree.ParseSize(size)
|
|
|
|
l.limit = value
|
|
|
|
l.input = size
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func validateLimits(min, max *LimitValue) bool {
|
|
|
|
if max.limit != 0 && min.limit > max.limit {
|
|
|
|
fmt.Fprintf(os.Stderr, "Given min-limit (%s) is above max-limit (%s)\n", min.input, max.input)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2020-03-08 21:43:01 +01:00
|
|
|
func min(a, b int) int {
|
|
|
|
if a < b {
|
|
|
|
return a
|
|
|
|
}
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
2020-03-01 23:09:00 +01:00
|
|
|
func main() {
|
|
|
|
var path string
|
|
|
|
var depth int
|
2020-03-07 17:23:06 +01:00
|
|
|
var root tree.Node
|
2020-03-08 20:58:13 +01:00
|
|
|
var interval time.Duration
|
2020-03-03 14:22:12 +01:00
|
|
|
var unit = &UnitValue{unit: "KB"}
|
2020-03-19 00:11:11 +01:00
|
|
|
var minLimit = &LimitValue{limit: 0}
|
|
|
|
var maxLimit = &LimitValue{limit: 0}
|
2020-03-08 20:58:13 +01:00
|
|
|
cli := flag.NewFlagSet(os.Args[0], flag.ExitOnError)
|
|
|
|
cli.IntVar(&depth, "depth", 0, "Depth to display")
|
|
|
|
cli.Var(unit, "unit", "Unit in which to report size")
|
2020-03-19 00:11:11 +01:00
|
|
|
cli.Var(minLimit, "limit-min", "Mininmum size of directories to display, with unit (0 means no limit)")
|
|
|
|
cli.Var(maxLimit, "limit-max", "Maximum size of directories to display, with unit (0 means no limit)")
|
2020-03-08 20:58:13 +01:00
|
|
|
cli.DurationVar(&interval, "interval", 100*time.Millisecond, "The update interval")
|
|
|
|
cli.Usage = func() {
|
|
|
|
fmt.Fprintf(cli.Output(), "Usage:\n%s [opts] [PATH]\n\n PATH: the root path to start from. Defaults to $PWD.\n\nopts:\n", os.Args[0])
|
|
|
|
cli.PrintDefaults()
|
|
|
|
}
|
|
|
|
cli.Parse(os.Args[1:])
|
|
|
|
|
2020-03-19 00:11:11 +01:00
|
|
|
if !validateLimits(minLimit, maxLimit) {
|
|
|
|
os.Exit(-1)
|
|
|
|
}
|
|
|
|
|
2020-03-08 20:58:13 +01:00
|
|
|
switch narg := cli.NArg(); narg {
|
|
|
|
case 0:
|
|
|
|
path = "."
|
|
|
|
case 1:
|
|
|
|
path = cli.Args()[0]
|
|
|
|
default:
|
|
|
|
fmt.Fprintln(os.Stderr, "Too many arguments")
|
|
|
|
os.Exit(-1)
|
|
|
|
}
|
2020-03-01 23:09:00 +01:00
|
|
|
|
2020-03-07 17:23:06 +01:00
|
|
|
if depth == 0 || !tree.AnyDirectoryDownThere(path) {
|
|
|
|
root = tree.NewSingle(path)
|
2020-03-05 14:32:50 +01:00
|
|
|
} else {
|
2020-03-07 17:23:06 +01:00
|
|
|
root = tree.NewTop(path)
|
2020-03-05 14:32:50 +01:00
|
|
|
}
|
2020-03-07 17:23:06 +01:00
|
|
|
root.SetUnit(unit.String())
|
2020-03-19 00:11:11 +01:00
|
|
|
root.SetLimits(minLimit.limit, maxLimit.limit)
|
2020-03-08 21:43:01 +01:00
|
|
|
out := log.New(os.Stdout, "", 0)
|
|
|
|
|
2020-03-07 17:23:06 +01:00
|
|
|
go root.Spawn(depth)
|
2020-03-08 21:43:01 +01:00
|
|
|
_, height, err := terminal.GetSize(int(os.Stdin.Fd()))
|
|
|
|
if err != nil {
|
|
|
|
fmt.Fprintln(os.Stderr, "Could not get terminal size")
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
2020-03-07 17:23:06 +01:00
|
|
|
|
2020-03-19 00:11:11 +01:00
|
|
|
treeDepth := 0
|
2020-03-07 17:23:06 +01:00
|
|
|
for {
|
2020-03-01 23:09:00 +01:00
|
|
|
select {
|
2020-03-08 20:58:13 +01:00
|
|
|
case <-time.After(interval):
|
2020-03-19 00:11:11 +01:00
|
|
|
if treeDepth != 0 {
|
|
|
|
out.Printf("\r\033[%dA", treeDepth)
|
2020-03-08 21:43:01 +01:00
|
|
|
} else {
|
|
|
|
out.Print("\r\033[A")
|
2020-03-08 20:58:13 +01:00
|
|
|
}
|
2020-03-19 00:11:11 +01:00
|
|
|
newTreeDepth := min(root.Depth()+2, height)
|
2020-03-08 21:43:01 +01:00
|
|
|
out.Print(root)
|
2020-03-19 00:11:11 +01:00
|
|
|
if newTreeDepth <= treeDepth {
|
|
|
|
//for i := treeDepth; i < newTreeDepth; i++ {
|
|
|
|
// out.Print("\n\033[2K")
|
|
|
|
//}
|
|
|
|
out.Print("\033[J\033[A")
|
|
|
|
}
|
|
|
|
treeDepth = newTreeDepth
|
2020-03-07 17:23:06 +01:00
|
|
|
if root.Complete() {
|
|
|
|
return
|
|
|
|
}
|
2020-03-01 23:09:00 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|