| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297 |
- // Copyright 2016 The etcd Authors
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- package command
- import (
- "encoding/json"
- "errors"
- "fmt"
- "os"
- "strings"
- v3 "github.com/coreos/etcd/clientv3"
- pb "github.com/coreos/etcd/etcdserver/etcdserverpb"
- spb "github.com/coreos/etcd/mvcc/mvccpb"
- "github.com/dustin/go-humanize"
- "github.com/olekukonko/tablewriter"
- )
- type printer interface {
- Del(v3.DeleteResponse)
- Get(v3.GetResponse)
- Put(v3.PutResponse)
- Txn(v3.TxnResponse)
- Watch(v3.WatchResponse)
- MemberList(v3.MemberListResponse)
- EndpointStatus([]epStatus)
- Alarm(v3.AlarmResponse)
- DBStatus(dbstatus)
- }
- func NewPrinter(printerType string, isHex bool) printer {
- switch printerType {
- case "simple":
- return &simplePrinter{isHex: isHex}
- case "json":
- return &jsonPrinter{}
- case "protobuf":
- return &pbPrinter{}
- case "table":
- return &tablePrinter{}
- }
- return nil
- }
- func makeMemberListTable(r v3.MemberListResponse) (hdr []string, rows [][]string) {
- hdr = []string{"ID", "Status", "Name", "Peer Addrs", "Client Addrs"}
- for _, m := range r.Members {
- status := "started"
- if len(m.Name) == 0 {
- status = "unstarted"
- }
- rows = append(rows, []string{
- fmt.Sprintf("%x", m.ID),
- status,
- m.Name,
- strings.Join(m.PeerURLs, ","),
- strings.Join(m.ClientURLs, ","),
- })
- }
- return
- }
- func makeEndpointStatusTable(statusList []epStatus) (hdr []string, rows [][]string) {
- hdr = []string{"endpoint", "ID", "version", "db size", "is leader", "raft term", "raft index"}
- for _, status := range statusList {
- rows = append(rows, []string{
- fmt.Sprint(status.Ep),
- fmt.Sprintf("%x", status.Resp.Header.MemberId),
- fmt.Sprint(status.Resp.Version),
- fmt.Sprint(humanize.Bytes(uint64(status.Resp.DbSize))),
- fmt.Sprint(status.Resp.Leader == status.Resp.Header.MemberId),
- fmt.Sprint(status.Resp.RaftTerm),
- fmt.Sprint(status.Resp.RaftIndex),
- })
- }
- return
- }
- func makeDBStatusTable(ds dbstatus) (hdr []string, rows [][]string) {
- hdr = []string{"hash", "revision", "total keys", "total size"}
- rows = append(rows, []string{
- fmt.Sprintf("%x", ds.Hash),
- fmt.Sprint(ds.Revision),
- fmt.Sprint(ds.TotalKey),
- humanize.Bytes(uint64(ds.TotalSize)),
- })
- return
- }
- type simplePrinter struct {
- isHex bool
- }
- func (s *simplePrinter) Del(resp v3.DeleteResponse) {
- fmt.Println(resp.Deleted)
- }
- func (s *simplePrinter) Get(resp v3.GetResponse) {
- for _, kv := range resp.Kvs {
- printKV(s.isHex, kv)
- }
- }
- func (s *simplePrinter) Put(r v3.PutResponse) { fmt.Println("OK") }
- func (s *simplePrinter) Txn(resp v3.TxnResponse) {
- if resp.Succeeded {
- fmt.Println("SUCCESS")
- } else {
- fmt.Println("FAILURE")
- }
- for _, r := range resp.Responses {
- fmt.Println("")
- switch v := r.Response.(type) {
- case *pb.ResponseUnion_ResponseDeleteRange:
- s.Del((v3.DeleteResponse)(*v.ResponseDeleteRange))
- case *pb.ResponseUnion_ResponsePut:
- s.Put((v3.PutResponse)(*v.ResponsePut))
- case *pb.ResponseUnion_ResponseRange:
- s.Get(((v3.GetResponse)(*v.ResponseRange)))
- default:
- fmt.Printf("unexpected response %+v\n", r)
- }
- }
- }
- func (s *simplePrinter) Watch(resp v3.WatchResponse) {
- for _, e := range resp.Events {
- fmt.Println(e.Type)
- printKV(s.isHex, e.Kv)
- }
- }
- func (s *simplePrinter) Alarm(resp v3.AlarmResponse) {
- for _, e := range resp.Alarms {
- fmt.Printf("%+v\n", e)
- }
- }
- func (s *simplePrinter) MemberList(resp v3.MemberListResponse) {
- _, rows := makeMemberListTable(resp)
- for _, row := range rows {
- fmt.Println(strings.Join(row, ", "))
- }
- }
- func (s *simplePrinter) EndpointStatus(statusList []epStatus) {
- _, rows := makeEndpointStatusTable(statusList)
- for _, row := range rows {
- fmt.Println(strings.Join(row, ", "))
- }
- }
- func (s *simplePrinter) DBStatus(ds dbstatus) {
- _, rows := makeDBStatusTable(ds)
- for _, row := range rows {
- fmt.Println(strings.Join(row, ", "))
- }
- }
- type tablePrinter struct{}
- func (tp *tablePrinter) Del(r v3.DeleteResponse) {
- ExitWithError(ExitBadFeature, errors.New("table is not supported as output format"))
- }
- func (tp *tablePrinter) Get(r v3.GetResponse) {
- ExitWithError(ExitBadFeature, errors.New("table is not supported as output format"))
- }
- func (tp *tablePrinter) Put(r v3.PutResponse) {
- ExitWithError(ExitBadFeature, errors.New("table is not supported as output format"))
- }
- func (tp *tablePrinter) Txn(r v3.TxnResponse) {
- ExitWithError(ExitBadFeature, errors.New("table is not supported as output format"))
- }
- func (tp *tablePrinter) Watch(r v3.WatchResponse) {
- ExitWithError(ExitBadFeature, errors.New("table is not supported as output format"))
- }
- func (tp *tablePrinter) Alarm(r v3.AlarmResponse) {
- ExitWithError(ExitBadFeature, errors.New("table is not supported as output format"))
- }
- func (tp *tablePrinter) MemberList(r v3.MemberListResponse) {
- hdr, rows := makeMemberListTable(r)
- table := tablewriter.NewWriter(os.Stdout)
- table.SetHeader(hdr)
- for _, row := range rows {
- table.Append(row)
- }
- table.Render()
- }
- func (tp *tablePrinter) EndpointStatus(r []epStatus) {
- hdr, rows := makeEndpointStatusTable(r)
- table := tablewriter.NewWriter(os.Stdout)
- table.SetHeader(hdr)
- for _, row := range rows {
- table.Append(row)
- }
- table.Render()
- }
- func (tp *tablePrinter) DBStatus(r dbstatus) {
- hdr, rows := makeDBStatusTable(r)
- table := tablewriter.NewWriter(os.Stdout)
- table.SetHeader(hdr)
- for _, row := range rows {
- table.Append(row)
- }
- table.Render()
- }
- type jsonPrinter struct{}
- func (p *jsonPrinter) Del(r v3.DeleteResponse) { printJSON(r) }
- func (p *jsonPrinter) Get(r v3.GetResponse) { printJSON(r) }
- func (p *jsonPrinter) Put(r v3.PutResponse) { printJSON(r) }
- func (p *jsonPrinter) Txn(r v3.TxnResponse) { printJSON(r) }
- func (p *jsonPrinter) Watch(r v3.WatchResponse) { printJSON(r) }
- func (p *jsonPrinter) Alarm(r v3.AlarmResponse) { printJSON(r) }
- func (p *jsonPrinter) MemberList(r v3.MemberListResponse) { printJSON(r) }
- func (p *jsonPrinter) EndpointStatus(r []epStatus) { printJSON(r) }
- func (p *jsonPrinter) DBStatus(r dbstatus) { printJSON(r) }
- func printJSON(v interface{}) {
- b, err := json.Marshal(v)
- if err != nil {
- fmt.Fprintf(os.Stderr, "%v\n", err)
- return
- }
- fmt.Println(string(b))
- }
- type pbPrinter struct{}
- type pbMarshal interface {
- Marshal() ([]byte, error)
- }
- func (p *pbPrinter) Del(r v3.DeleteResponse) {
- printPB((*pb.DeleteRangeResponse)(&r))
- }
- func (p *pbPrinter) Get(r v3.GetResponse) {
- printPB((*pb.RangeResponse)(&r))
- }
- func (p *pbPrinter) Put(r v3.PutResponse) {
- printPB((*pb.PutResponse)(&r))
- }
- func (p *pbPrinter) Txn(r v3.TxnResponse) {
- printPB((*pb.TxnResponse)(&r))
- }
- func (p *pbPrinter) Watch(r v3.WatchResponse) {
- for _, ev := range r.Events {
- printPB((*spb.Event)(ev))
- }
- }
- func (p *pbPrinter) Alarm(r v3.AlarmResponse) {
- printPB((*pb.AlarmResponse)(&r))
- }
- func (pb *pbPrinter) MemberList(r v3.MemberListResponse) {
- ExitWithError(ExitBadFeature, errors.New("only support simple or json as output format"))
- }
- func (pb *pbPrinter) EndpointStatus(statusList []epStatus) {
- ExitWithError(ExitBadFeature, errors.New("only support simple or json as output format"))
- }
- func (pb *pbPrinter) DBStatus(r dbstatus) {
- ExitWithError(ExitBadFeature, errors.New("only support simple or json as output format"))
- }
- func printPB(m pbMarshal) {
- b, err := m.Marshal()
- if err != nil {
- fmt.Fprintf(os.Stderr, "%v\n", err)
- return
- }
- fmt.Printf(string(b))
- }
|