printer.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457
  1. // Copyright 2016 The etcd Authors
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. package command
  15. import (
  16. "encoding/json"
  17. "errors"
  18. "fmt"
  19. "os"
  20. "strings"
  21. v3 "github.com/coreos/etcd/clientv3"
  22. pb "github.com/coreos/etcd/etcdserver/etcdserverpb"
  23. spb "github.com/coreos/etcd/mvcc/mvccpb"
  24. "github.com/dustin/go-humanize"
  25. "github.com/olekukonko/tablewriter"
  26. )
  27. type printer interface {
  28. Del(v3.DeleteResponse)
  29. Get(v3.GetResponse)
  30. Put(v3.PutResponse)
  31. Txn(v3.TxnResponse)
  32. Watch(v3.WatchResponse)
  33. TimeToLive(r v3.LeaseTimeToLiveResponse, keys bool)
  34. MemberList(v3.MemberListResponse)
  35. EndpointStatus([]epStatus)
  36. Alarm(v3.AlarmResponse)
  37. DBStatus(dbstatus)
  38. }
  39. func NewPrinter(printerType string, isHex bool) printer {
  40. switch printerType {
  41. case "simple":
  42. return &simplePrinter{isHex: isHex}
  43. case "fields":
  44. return &fieldsPrinter{}
  45. case "json":
  46. return &jsonPrinter{}
  47. case "protobuf":
  48. return &pbPrinter{}
  49. case "table":
  50. return &tablePrinter{}
  51. }
  52. return nil
  53. }
  54. func makeMemberListTable(r v3.MemberListResponse) (hdr []string, rows [][]string) {
  55. hdr = []string{"ID", "Status", "Name", "Peer Addrs", "Client Addrs"}
  56. for _, m := range r.Members {
  57. status := "started"
  58. if len(m.Name) == 0 {
  59. status = "unstarted"
  60. }
  61. rows = append(rows, []string{
  62. fmt.Sprintf("%x", m.ID),
  63. status,
  64. m.Name,
  65. strings.Join(m.PeerURLs, ","),
  66. strings.Join(m.ClientURLs, ","),
  67. })
  68. }
  69. return
  70. }
  71. func makeEndpointStatusTable(statusList []epStatus) (hdr []string, rows [][]string) {
  72. hdr = []string{"endpoint", "ID", "version", "db size", "is leader", "raft term", "raft index"}
  73. for _, status := range statusList {
  74. rows = append(rows, []string{
  75. fmt.Sprint(status.Ep),
  76. fmt.Sprintf("%x", status.Resp.Header.MemberId),
  77. fmt.Sprint(status.Resp.Version),
  78. fmt.Sprint(humanize.Bytes(uint64(status.Resp.DbSize))),
  79. fmt.Sprint(status.Resp.Leader == status.Resp.Header.MemberId),
  80. fmt.Sprint(status.Resp.RaftTerm),
  81. fmt.Sprint(status.Resp.RaftIndex),
  82. })
  83. }
  84. return
  85. }
  86. func makeDBStatusTable(ds dbstatus) (hdr []string, rows [][]string) {
  87. hdr = []string{"hash", "revision", "total keys", "total size"}
  88. rows = append(rows, []string{
  89. fmt.Sprintf("%x", ds.Hash),
  90. fmt.Sprint(ds.Revision),
  91. fmt.Sprint(ds.TotalKey),
  92. humanize.Bytes(uint64(ds.TotalSize)),
  93. })
  94. return
  95. }
  96. type simplePrinter struct {
  97. isHex bool
  98. valueOnly bool
  99. }
  100. func (s *simplePrinter) Del(resp v3.DeleteResponse) {
  101. fmt.Println(resp.Deleted)
  102. for _, kv := range resp.PrevKvs {
  103. printKV(s.isHex, s.valueOnly, kv)
  104. }
  105. }
  106. func (s *simplePrinter) Get(resp v3.GetResponse) {
  107. for _, kv := range resp.Kvs {
  108. printKV(s.isHex, s.valueOnly, kv)
  109. }
  110. }
  111. func (s *simplePrinter) Put(r v3.PutResponse) {
  112. fmt.Println("OK")
  113. if r.PrevKv != nil {
  114. printKV(s.isHex, s.valueOnly, r.PrevKv)
  115. }
  116. }
  117. func (s *simplePrinter) Txn(resp v3.TxnResponse) {
  118. if resp.Succeeded {
  119. fmt.Println("SUCCESS")
  120. } else {
  121. fmt.Println("FAILURE")
  122. }
  123. for _, r := range resp.Responses {
  124. fmt.Println("")
  125. switch v := r.Response.(type) {
  126. case *pb.ResponseOp_ResponseDeleteRange:
  127. s.Del((v3.DeleteResponse)(*v.ResponseDeleteRange))
  128. case *pb.ResponseOp_ResponsePut:
  129. s.Put((v3.PutResponse)(*v.ResponsePut))
  130. case *pb.ResponseOp_ResponseRange:
  131. s.Get(((v3.GetResponse)(*v.ResponseRange)))
  132. default:
  133. fmt.Printf("unexpected response %+v\n", r)
  134. }
  135. }
  136. }
  137. func (s *simplePrinter) Watch(resp v3.WatchResponse) {
  138. for _, e := range resp.Events {
  139. fmt.Println(e.Type)
  140. if e.PrevKv != nil {
  141. printKV(s.isHex, s.valueOnly, e.PrevKv)
  142. }
  143. printKV(s.isHex, s.valueOnly, e.Kv)
  144. }
  145. }
  146. func (s *simplePrinter) TimeToLive(resp v3.LeaseTimeToLiveResponse, keys bool) {
  147. txt := fmt.Sprintf("lease %016x granted with TTL(%ds), remaining(%ds)", resp.ID, resp.GrantedTTL, resp.TTL)
  148. if keys {
  149. ks := make([]string, len(resp.Keys))
  150. for i := range resp.Keys {
  151. ks[i] = string(resp.Keys[i])
  152. }
  153. txt += fmt.Sprintf(", attached keys(%v)", ks)
  154. }
  155. fmt.Println(txt)
  156. }
  157. func (s *simplePrinter) Alarm(resp v3.AlarmResponse) {
  158. for _, e := range resp.Alarms {
  159. fmt.Printf("%+v\n", e)
  160. }
  161. }
  162. func (s *simplePrinter) MemberList(resp v3.MemberListResponse) {
  163. _, rows := makeMemberListTable(resp)
  164. for _, row := range rows {
  165. fmt.Println(strings.Join(row, ", "))
  166. }
  167. }
  168. func (s *simplePrinter) EndpointStatus(statusList []epStatus) {
  169. _, rows := makeEndpointStatusTable(statusList)
  170. for _, row := range rows {
  171. fmt.Println(strings.Join(row, ", "))
  172. }
  173. }
  174. func (s *simplePrinter) DBStatus(ds dbstatus) {
  175. _, rows := makeDBStatusTable(ds)
  176. for _, row := range rows {
  177. fmt.Println(strings.Join(row, ", "))
  178. }
  179. }
  180. type tablePrinter struct{}
  181. func (tp *tablePrinter) Del(r v3.DeleteResponse) {
  182. ExitWithError(ExitBadFeature, errors.New("table is not supported as output format"))
  183. }
  184. func (tp *tablePrinter) Get(r v3.GetResponse) {
  185. ExitWithError(ExitBadFeature, errors.New("table is not supported as output format"))
  186. }
  187. func (tp *tablePrinter) Put(r v3.PutResponse) {
  188. ExitWithError(ExitBadFeature, errors.New("table is not supported as output format"))
  189. }
  190. func (tp *tablePrinter) Txn(r v3.TxnResponse) {
  191. ExitWithError(ExitBadFeature, errors.New("table is not supported as output format"))
  192. }
  193. func (tp *tablePrinter) Watch(r v3.WatchResponse) {
  194. ExitWithError(ExitBadFeature, errors.New("table is not supported as output format"))
  195. }
  196. func (tp *tablePrinter) TimeToLive(r v3.LeaseTimeToLiveResponse, keys bool) {
  197. ExitWithError(ExitBadFeature, errors.New("table is not supported as output format"))
  198. }
  199. func (tp *tablePrinter) Alarm(r v3.AlarmResponse) {
  200. ExitWithError(ExitBadFeature, errors.New("table is not supported as output format"))
  201. }
  202. func (tp *tablePrinter) MemberList(r v3.MemberListResponse) {
  203. hdr, rows := makeMemberListTable(r)
  204. table := tablewriter.NewWriter(os.Stdout)
  205. table.SetHeader(hdr)
  206. for _, row := range rows {
  207. table.Append(row)
  208. }
  209. table.Render()
  210. }
  211. func (tp *tablePrinter) EndpointStatus(r []epStatus) {
  212. hdr, rows := makeEndpointStatusTable(r)
  213. table := tablewriter.NewWriter(os.Stdout)
  214. table.SetHeader(hdr)
  215. for _, row := range rows {
  216. table.Append(row)
  217. }
  218. table.Render()
  219. }
  220. func (tp *tablePrinter) DBStatus(r dbstatus) {
  221. hdr, rows := makeDBStatusTable(r)
  222. table := tablewriter.NewWriter(os.Stdout)
  223. table.SetHeader(hdr)
  224. for _, row := range rows {
  225. table.Append(row)
  226. }
  227. table.Render()
  228. }
  229. type jsonPrinter struct{}
  230. func (p *jsonPrinter) Del(r v3.DeleteResponse) { printJSON(r) }
  231. func (p *jsonPrinter) Get(r v3.GetResponse) { printJSON(r) }
  232. func (p *jsonPrinter) Put(r v3.PutResponse) { printJSON(r) }
  233. func (p *jsonPrinter) Txn(r v3.TxnResponse) { printJSON(r) }
  234. func (p *jsonPrinter) Watch(r v3.WatchResponse) { printJSON(r) }
  235. func (p *jsonPrinter) TimeToLive(r v3.LeaseTimeToLiveResponse, keys bool) { printJSON(r) }
  236. func (p *jsonPrinter) Alarm(r v3.AlarmResponse) { printJSON(r) }
  237. func (p *jsonPrinter) MemberList(r v3.MemberListResponse) { printJSON(r) }
  238. func (p *jsonPrinter) EndpointStatus(r []epStatus) { printJSON(r) }
  239. func (p *jsonPrinter) DBStatus(r dbstatus) { printJSON(r) }
  240. func printJSON(v interface{}) {
  241. b, err := json.Marshal(v)
  242. if err != nil {
  243. fmt.Fprintf(os.Stderr, "%v\n", err)
  244. return
  245. }
  246. fmt.Println(string(b))
  247. }
  248. type pbPrinter struct{}
  249. type pbMarshal interface {
  250. Marshal() ([]byte, error)
  251. }
  252. func (p *pbPrinter) Del(r v3.DeleteResponse) {
  253. printPB((*pb.DeleteRangeResponse)(&r))
  254. }
  255. func (p *pbPrinter) Get(r v3.GetResponse) {
  256. printPB((*pb.RangeResponse)(&r))
  257. }
  258. func (p *pbPrinter) Put(r v3.PutResponse) {
  259. printPB((*pb.PutResponse)(&r))
  260. }
  261. func (p *pbPrinter) Txn(r v3.TxnResponse) {
  262. printPB((*pb.TxnResponse)(&r))
  263. }
  264. func (p *pbPrinter) Watch(r v3.WatchResponse) {
  265. for _, ev := range r.Events {
  266. printPB((*spb.Event)(ev))
  267. }
  268. }
  269. func (p *pbPrinter) TimeToLive(r v3.LeaseTimeToLiveResponse, keys bool) {
  270. ExitWithError(ExitBadFeature, errors.New("only support simple or json as output format"))
  271. }
  272. func (p *pbPrinter) Alarm(r v3.AlarmResponse) {
  273. printPB((*pb.AlarmResponse)(&r))
  274. }
  275. func (p *pbPrinter) MemberList(r v3.MemberListResponse) {
  276. printPB((*pb.MemberListResponse)(&r))
  277. }
  278. func (p *pbPrinter) EndpointStatus(statusList []epStatus) {
  279. ExitWithError(ExitBadFeature, errors.New("only support simple or json as output format"))
  280. }
  281. func (p *pbPrinter) DBStatus(r dbstatus) {
  282. ExitWithError(ExitBadFeature, errors.New("only support simple or json as output format"))
  283. }
  284. func printPB(m pbMarshal) {
  285. b, err := m.Marshal()
  286. if err != nil {
  287. fmt.Fprintf(os.Stderr, "%v\n", err)
  288. return
  289. }
  290. fmt.Printf(string(b))
  291. }
  292. type fieldsPrinter struct{}
  293. func (p *fieldsPrinter) kv(pfx string, kv *spb.KeyValue) {
  294. fmt.Printf("\"%sKey\" : %q\n", pfx, string(kv.Key))
  295. fmt.Printf("\"%sCreateRevision\" : %d\n", pfx, kv.CreateRevision)
  296. fmt.Printf("\"%sModRevision\" : %d\n", pfx, kv.ModRevision)
  297. fmt.Printf("\"%sVersion\" : %d\n", pfx, kv.Version)
  298. fmt.Printf("\"%sValue\" : %q\n", pfx, string(kv.Value))
  299. fmt.Printf("\"%sLease\" : %d\n", pfx, string(kv.Lease))
  300. }
  301. func (p *fieldsPrinter) hdr(h *pb.ResponseHeader) {
  302. fmt.Println(`"ClusterID" :`, h.ClusterId)
  303. fmt.Println(`"MemberID" :`, h.MemberId)
  304. fmt.Println(`"Revision" :`, h.Revision)
  305. fmt.Println(`"RaftTerm" :`, h.RaftTerm)
  306. }
  307. func (p *fieldsPrinter) Del(r v3.DeleteResponse) {
  308. p.hdr(r.Header)
  309. fmt.Println(`"Deleted" :`, r.Deleted)
  310. for _, kv := range r.PrevKvs {
  311. p.kv("Prev", kv)
  312. }
  313. }
  314. func (p *fieldsPrinter) Get(r v3.GetResponse) {
  315. p.hdr(r.Header)
  316. for _, kv := range r.Kvs {
  317. p.kv("", kv)
  318. }
  319. fmt.Println(`"More" :`, r.More)
  320. fmt.Println(`"Count" :`, r.Count)
  321. }
  322. func (p *fieldsPrinter) Put(r v3.PutResponse) {
  323. p.hdr(r.Header)
  324. if r.PrevKv != nil {
  325. p.kv("Prev", r.PrevKv)
  326. }
  327. }
  328. func (p *fieldsPrinter) Txn(r v3.TxnResponse) {
  329. p.hdr(r.Header)
  330. fmt.Println(`"Succeeded" :`, r.Succeeded)
  331. for _, resp := range r.Responses {
  332. switch v := resp.Response.(type) {
  333. case *pb.ResponseOp_ResponseDeleteRange:
  334. p.Del((v3.DeleteResponse)(*v.ResponseDeleteRange))
  335. case *pb.ResponseOp_ResponsePut:
  336. p.Put((v3.PutResponse)(*v.ResponsePut))
  337. case *pb.ResponseOp_ResponseRange:
  338. p.Get((v3.GetResponse)(*v.ResponseRange))
  339. default:
  340. fmt.Printf("\"Unknown\" : %q\n", fmt.Sprintf("%+v", v))
  341. }
  342. }
  343. }
  344. func (p *fieldsPrinter) Watch(resp v3.WatchResponse) {
  345. p.hdr(&resp.Header)
  346. for _, e := range resp.Events {
  347. fmt.Println(`"Type" : `, e.Type)
  348. if e.PrevKv != nil {
  349. p.kv("Prev", e.PrevKv)
  350. }
  351. p.kv("", e.Kv)
  352. }
  353. }
  354. func (p *fieldsPrinter) TimeToLive(r v3.LeaseTimeToLiveResponse, keys bool) {
  355. p.hdr(r.ResponseHeader)
  356. fmt.Println(`"ID" :`, r.ID)
  357. fmt.Println(`"TTL" :`, r.TTL)
  358. fmt.Println(`"GrantedTTL" :`, r.GrantedTTL)
  359. for _, k := range r.Keys {
  360. fmt.Printf("\"Key\" : %q\n", string(k))
  361. }
  362. }
  363. func (p *fieldsPrinter) MemberList(r v3.MemberListResponse) {
  364. p.hdr(r.Header)
  365. for _, m := range r.Members {
  366. fmt.Println(`"ID" :`, m.ID)
  367. fmt.Printf("\"Name\" : %q\n", m.Name)
  368. for _, u := range m.PeerURLs {
  369. fmt.Printf("\"PeerURL\" : %q\n", u)
  370. }
  371. for _, u := range m.ClientURLs {
  372. fmt.Printf("\"ClientURL\" : %q\n", u)
  373. }
  374. fmt.Println()
  375. }
  376. }
  377. func (p *fieldsPrinter) EndpointStatus(eps []epStatus) {
  378. for _, ep := range eps {
  379. p.hdr(ep.Resp.Header)
  380. fmt.Printf("\"Version\" : %q\n", ep.Resp.Version)
  381. fmt.Println(`"DBSize" :"`, ep.Resp.DbSize)
  382. fmt.Println(`"Leader" :"`, ep.Resp.Leader)
  383. fmt.Println(`"RaftIndex" :"`, ep.Resp.RaftIndex)
  384. fmt.Println(`"RaftTerm" :"`, ep.Resp.RaftTerm)
  385. fmt.Printf("\"Endpoint\" : %q\n", ep.Ep)
  386. fmt.Println()
  387. }
  388. }
  389. func (p *fieldsPrinter) Alarm(r v3.AlarmResponse) {
  390. p.hdr(r.Header)
  391. for _, a := range r.Alarms {
  392. fmt.Println(`"MemberID" :`, a.MemberID)
  393. fmt.Println(`"AlarmType" :`, a.Alarm)
  394. fmt.Println()
  395. }
  396. }
  397. func (p *fieldsPrinter) DBStatus(r dbstatus) {
  398. fmt.Println(`"Hash" :`, r.Hash)
  399. fmt.Println(`"Revision" :`, r.Revision)
  400. fmt.Println(`"Keys" :`, r.TotalKey)
  401. fmt.Println(`"Size" :`, r.TotalSize)
  402. }