mirror of
https://github.com/zrepl/zrepl.git
synced 2024-11-28 19:34:58 +01:00
101 lines
2.1 KiB
Go
101 lines
2.1 KiB
Go
|
package client
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"encoding/json"
|
||
|
"fmt"
|
||
|
"os"
|
||
|
"sync"
|
||
|
|
||
|
"github.com/fatih/color"
|
||
|
"github.com/pkg/errors"
|
||
|
"github.com/spf13/pflag"
|
||
|
|
||
|
"github.com/zrepl/zrepl/cli"
|
||
|
"github.com/zrepl/zrepl/endpoint"
|
||
|
"github.com/zrepl/zrepl/util/chainlock"
|
||
|
)
|
||
|
|
||
|
var zabsListFlags struct {
|
||
|
Filter zabsFilterFlags
|
||
|
Json bool
|
||
|
}
|
||
|
|
||
|
var zabsCmdList = &cli.Subcommand{
|
||
|
Use: "list",
|
||
|
Short: `list zrepl ZFS abstractions`,
|
||
|
Run: doZabsList,
|
||
|
NoRequireConfig: true,
|
||
|
SetupFlags: func(f *pflag.FlagSet) {
|
||
|
zabsListFlags.Filter.registerZabsFilterFlags(f, "list")
|
||
|
f.BoolVar(&zabsListFlags.Json, "json", false, "emit JSON")
|
||
|
},
|
||
|
}
|
||
|
|
||
|
func doZabsList(sc *cli.Subcommand, args []string) error {
|
||
|
var err error
|
||
|
ctx := context.Background()
|
||
|
|
||
|
if len(args) > 0 {
|
||
|
return errors.New("this subcommand takes no positional arguments")
|
||
|
}
|
||
|
|
||
|
q, err := zabsListFlags.Filter.Query()
|
||
|
if err != nil {
|
||
|
return errors.Wrap(err, "invalid filter specification on command line")
|
||
|
}
|
||
|
|
||
|
abstractions, errors, err := endpoint.ListAbstractionsStreamed(ctx, q)
|
||
|
if err != nil {
|
||
|
return err // context clear by invocation of command
|
||
|
}
|
||
|
|
||
|
var line chainlock.L
|
||
|
var wg sync.WaitGroup
|
||
|
defer wg.Wait()
|
||
|
wg.Add(1)
|
||
|
|
||
|
// print results
|
||
|
go func() {
|
||
|
defer wg.Done()
|
||
|
enc := json.NewEncoder(os.Stdout)
|
||
|
for a := range abstractions {
|
||
|
func() {
|
||
|
defer line.Lock().Unlock()
|
||
|
if zabsListFlags.Json {
|
||
|
enc.SetIndent("", " ")
|
||
|
if err := enc.Encode(abstractions); err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
fmt.Println()
|
||
|
} else {
|
||
|
fmt.Println(a)
|
||
|
}
|
||
|
}()
|
||
|
}
|
||
|
}()
|
||
|
|
||
|
// print errors to stderr
|
||
|
errorColor := color.New(color.FgRed)
|
||
|
var errorsSlice []endpoint.ListAbstractionsError
|
||
|
wg.Add(1)
|
||
|
go func() {
|
||
|
defer wg.Done()
|
||
|
for err := range errors {
|
||
|
func() {
|
||
|
defer line.Lock().Unlock()
|
||
|
errorsSlice = append(errorsSlice, err)
|
||
|
errorColor.Fprintf(os.Stderr, "%s\n", err)
|
||
|
}()
|
||
|
}
|
||
|
}()
|
||
|
wg.Wait()
|
||
|
if len(errorsSlice) > 0 {
|
||
|
errorColor.Add(color.Bold).Fprintf(os.Stderr, "there were errors in listing the abstractions")
|
||
|
return fmt.Errorf("")
|
||
|
} else {
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
}
|