Merge pull request #913 from csrwng/pluggable_type_printing
Allow kubecfg to print custom types
This commit is contained in:
@@ -311,7 +311,7 @@ func executeAPIRequest(method string, s *kube_client.Client) bool {
|
|||||||
Template: tmpl,
|
Template: tmpl,
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
printer = &kubecfg.HumanReadablePrinter{}
|
printer = humanReadablePrinter()
|
||||||
}
|
}
|
||||||
|
|
||||||
if err = printer.PrintObj(obj, os.Stdout); err != nil {
|
if err = printer.PrintObj(obj, os.Stdout); err != nil {
|
||||||
@@ -369,3 +369,9 @@ func executeControllerRequest(method string, c *kube_client.Client) bool {
|
|||||||
}
|
}
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func humanReadablePrinter() *kubecfg.HumanReadablePrinter {
|
||||||
|
printer := kubecfg.NewHumanReadablePrinter()
|
||||||
|
// Add Handler calls here to support additional types
|
||||||
|
return printer
|
||||||
|
}
|
||||||
|
@@ -20,12 +20,14 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
|
"reflect"
|
||||||
"strings"
|
"strings"
|
||||||
"text/tabwriter"
|
"text/tabwriter"
|
||||||
"text/template"
|
"text/template"
|
||||||
|
|
||||||
"github.com/GoogleCloudPlatform/kubernetes/pkg/api"
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/api"
|
||||||
"github.com/GoogleCloudPlatform/kubernetes/pkg/labels"
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/labels"
|
||||||
|
"github.com/golang/glog"
|
||||||
"gopkg.in/v1/yaml"
|
"gopkg.in/v1/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -81,8 +83,58 @@ func (y *YAMLPrinter) PrintObj(obj interface{}, w io.Writer) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type handlerEntry struct {
|
||||||
|
columns []string
|
||||||
|
printFunc reflect.Value
|
||||||
|
}
|
||||||
|
|
||||||
// HumanReadablePrinter is an implementation of ResourcePrinter which attempts to provide more elegant output.
|
// HumanReadablePrinter is an implementation of ResourcePrinter which attempts to provide more elegant output.
|
||||||
type HumanReadablePrinter struct{}
|
type HumanReadablePrinter struct {
|
||||||
|
handlerMap map[reflect.Type]*handlerEntry
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewHumanReadablePrinter creates a HumanReadablePrinter
|
||||||
|
func NewHumanReadablePrinter() *HumanReadablePrinter {
|
||||||
|
printer := &HumanReadablePrinter{make(map[reflect.Type]*handlerEntry)}
|
||||||
|
printer.addDefaultHandlers()
|
||||||
|
return printer
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handler adds a print handler with a given set of columns to HumanReadablePrinter instance
|
||||||
|
// printFunc is the function that will be called to print an object
|
||||||
|
// It must be of the following type:
|
||||||
|
// func printFunc(object ObjectType, w io.Writer) error
|
||||||
|
// where ObjectType is the type of the object that will be printed.
|
||||||
|
func (h *HumanReadablePrinter) Handler(columns []string, printFunc interface{}) error {
|
||||||
|
printFuncValue := reflect.ValueOf(printFunc)
|
||||||
|
if err := h.validatePrintHandlerFunc(printFuncValue); err != nil {
|
||||||
|
glog.Errorf("Unable to add print handler: %v", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
objType := printFuncValue.Type().In(0)
|
||||||
|
h.handlerMap[objType] = &handlerEntry{
|
||||||
|
columns: columns,
|
||||||
|
printFunc: printFuncValue,
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *HumanReadablePrinter) validatePrintHandlerFunc(printFunc reflect.Value) error {
|
||||||
|
if printFunc.Kind() != reflect.Func {
|
||||||
|
return fmt.Errorf("Invalid print handler. %#v is not a function.", printFunc)
|
||||||
|
}
|
||||||
|
funcType := printFunc.Type()
|
||||||
|
if funcType.NumIn() != 2 || funcType.NumOut() != 1 {
|
||||||
|
return fmt.Errorf("Invalid print handler." +
|
||||||
|
"Must accept 2 parameters and return 1 value.")
|
||||||
|
}
|
||||||
|
if funcType.In(1) != reflect.TypeOf((*io.Writer)(nil)).Elem() ||
|
||||||
|
funcType.Out(0) != reflect.TypeOf((*error)(nil)).Elem() {
|
||||||
|
return fmt.Errorf("Invalid print handler. The expected signature is: "+
|
||||||
|
"func handler(obj %v, w io.Writer) error", funcType.In(0))
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
var podColumns = []string{"Name", "Image(s)", "Host", "Labels"}
|
var podColumns = []string{"Name", "Image(s)", "Host", "Labels"}
|
||||||
var replicationControllerColumns = []string{"Name", "Image(s)", "Selector", "Replicas"}
|
var replicationControllerColumns = []string{"Name", "Image(s)", "Selector", "Replicas"}
|
||||||
@@ -90,6 +142,19 @@ var serviceColumns = []string{"Name", "Labels", "Selector", "Port"}
|
|||||||
var minionColumns = []string{"Minion identifier"}
|
var minionColumns = []string{"Minion identifier"}
|
||||||
var statusColumns = []string{"Status"}
|
var statusColumns = []string{"Status"}
|
||||||
|
|
||||||
|
// handleDefaultTypes adds print handlers for default Kubernetes types
|
||||||
|
func (h *HumanReadablePrinter) addDefaultHandlers() {
|
||||||
|
h.Handler(podColumns, printPod)
|
||||||
|
h.Handler(podColumns, printPodList)
|
||||||
|
h.Handler(replicationControllerColumns, printReplicationController)
|
||||||
|
h.Handler(replicationControllerColumns, printReplicationControllerList)
|
||||||
|
h.Handler(serviceColumns, printService)
|
||||||
|
h.Handler(serviceColumns, printServiceList)
|
||||||
|
h.Handler(minionColumns, printMinion)
|
||||||
|
h.Handler(minionColumns, printMinionList)
|
||||||
|
h.Handler(statusColumns, printStatus)
|
||||||
|
}
|
||||||
|
|
||||||
func (h *HumanReadablePrinter) unknown(data []byte, w io.Writer) error {
|
func (h *HumanReadablePrinter) unknown(data []byte, w io.Writer) error {
|
||||||
_, err := fmt.Fprintf(w, "Unknown object: %s", string(data))
|
_, err := fmt.Fprintf(w, "Unknown object: %s", string(data))
|
||||||
return err
|
return err
|
||||||
@@ -107,7 +172,7 @@ func (h *HumanReadablePrinter) printHeader(columnNames []string, w io.Writer) er
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *HumanReadablePrinter) makeImageList(manifest api.ContainerManifest) string {
|
func makeImageList(manifest api.ContainerManifest) string {
|
||||||
var images []string
|
var images []string
|
||||||
for _, container := range manifest.Containers {
|
for _, container := range manifest.Containers {
|
||||||
images = append(images, container.Image)
|
images = append(images, container.Image)
|
||||||
@@ -115,74 +180,74 @@ func (h *HumanReadablePrinter) makeImageList(manifest api.ContainerManifest) str
|
|||||||
return strings.Join(images, ",")
|
return strings.Join(images, ",")
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *HumanReadablePrinter) printPod(pod *api.Pod, w io.Writer) error {
|
func printPod(pod *api.Pod, w io.Writer) error {
|
||||||
_, err := fmt.Fprintf(w, "%s\t%s\t%s\t%s\n",
|
_, err := fmt.Fprintf(w, "%s\t%s\t%s\t%s\n",
|
||||||
pod.ID, h.makeImageList(pod.DesiredState.Manifest), pod.CurrentState.Host+"/"+pod.CurrentState.HostIP, labels.Set(pod.Labels))
|
pod.ID, makeImageList(pod.DesiredState.Manifest),
|
||||||
|
pod.CurrentState.Host+"/"+pod.CurrentState.HostIP, labels.Set(pod.Labels))
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *HumanReadablePrinter) printPodList(podList *api.PodList, w io.Writer) error {
|
func printPodList(podList *api.PodList, w io.Writer) error {
|
||||||
for _, pod := range podList.Items {
|
for _, pod := range podList.Items {
|
||||||
if err := h.printPod(&pod, w); err != nil {
|
if err := printPod(&pod, w); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *HumanReadablePrinter) printReplicationController(ctrl *api.ReplicationController, w io.Writer) error {
|
func printReplicationController(ctrl *api.ReplicationController, w io.Writer) error {
|
||||||
_, err := fmt.Fprintf(w, "%s\t%s\t%s\t%d\n",
|
_, err := fmt.Fprintf(w, "%s\t%s\t%s\t%d\n",
|
||||||
ctrl.ID, h.makeImageList(ctrl.DesiredState.PodTemplate.DesiredState.Manifest), labels.Set(ctrl.DesiredState.ReplicaSelector), ctrl.DesiredState.Replicas)
|
ctrl.ID, makeImageList(ctrl.DesiredState.PodTemplate.DesiredState.Manifest),
|
||||||
|
labels.Set(ctrl.DesiredState.ReplicaSelector), ctrl.DesiredState.Replicas)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *HumanReadablePrinter) printReplicationControllerList(list *api.ReplicationControllerList, w io.Writer) error {
|
func printReplicationControllerList(list *api.ReplicationControllerList, w io.Writer) error {
|
||||||
for _, ctrl := range list.Items {
|
for _, ctrl := range list.Items {
|
||||||
if err := h.printReplicationController(&ctrl, w); err != nil {
|
if err := printReplicationController(&ctrl, w); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *HumanReadablePrinter) printService(svc *api.Service, w io.Writer) error {
|
func printService(svc *api.Service, w io.Writer) error {
|
||||||
_, err := fmt.Fprintf(w, "%s\t%s\t%s\t%d\n", svc.ID, labels.Set(svc.Labels), labels.Set(svc.Selector), svc.Port)
|
_, err := fmt.Fprintf(w, "%s\t%s\t%s\t%d\n", svc.ID, labels.Set(svc.Labels),
|
||||||
|
labels.Set(svc.Selector), svc.Port)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *HumanReadablePrinter) printServiceList(list *api.ServiceList, w io.Writer) error {
|
func printServiceList(list *api.ServiceList, w io.Writer) error {
|
||||||
for _, svc := range list.Items {
|
for _, svc := range list.Items {
|
||||||
if err := h.printService(&svc, w); err != nil {
|
if err := printService(&svc, w); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *HumanReadablePrinter) printMinion(minion *api.Minion, w io.Writer) error {
|
func printMinion(minion *api.Minion, w io.Writer) error {
|
||||||
_, err := fmt.Fprintf(w, "%s\n", minion.ID)
|
_, err := fmt.Fprintf(w, "%s\n", minion.ID)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *HumanReadablePrinter) printMinionList(list *api.MinionList, w io.Writer) error {
|
func printMinionList(list *api.MinionList, w io.Writer) error {
|
||||||
for _, minion := range list.Items {
|
for _, minion := range list.Items {
|
||||||
if err := h.printMinion(&minion, w); err != nil {
|
if err := printMinion(&minion, w); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *HumanReadablePrinter) printStatus(status *api.Status, w io.Writer) error {
|
func printStatus(status *api.Status, w io.Writer) error {
|
||||||
err := h.printHeader(statusColumns, w)
|
_, err := fmt.Fprintf(w, "%v\n", status.Status)
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
_, err = fmt.Fprintf(w, "%v\n", status.Status)
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Print parses the data as JSON, then prints the parsed data in a human-friendly format according to the type of the data.
|
// Print parses the data as JSON, then prints the parsed data in a human-friendly
|
||||||
|
// format according to the type of the data.
|
||||||
func (h *HumanReadablePrinter) Print(data []byte, output io.Writer) error {
|
func (h *HumanReadablePrinter) Print(data []byte, output io.Writer) error {
|
||||||
var mapObj map[string]interface{}
|
var mapObj map[string]interface{}
|
||||||
if err := json.Unmarshal([]byte(data), &mapObj); err != nil {
|
if err := json.Unmarshal([]byte(data), &mapObj); err != nil {
|
||||||
@@ -204,36 +269,17 @@ func (h *HumanReadablePrinter) Print(data []byte, output io.Writer) error {
|
|||||||
func (h *HumanReadablePrinter) PrintObj(obj interface{}, output io.Writer) error {
|
func (h *HumanReadablePrinter) PrintObj(obj interface{}, output io.Writer) error {
|
||||||
w := tabwriter.NewWriter(output, 20, 5, 3, ' ', 0)
|
w := tabwriter.NewWriter(output, 20, 5, 3, ' ', 0)
|
||||||
defer w.Flush()
|
defer w.Flush()
|
||||||
switch o := obj.(type) {
|
if handler := h.handlerMap[reflect.TypeOf(obj)]; handler != nil {
|
||||||
case *api.Pod:
|
h.printHeader(handler.columns, w)
|
||||||
h.printHeader(podColumns, w)
|
args := []reflect.Value{reflect.ValueOf(obj), reflect.ValueOf(w)}
|
||||||
return h.printPod(o, w)
|
resultValue := handler.printFunc.Call(args)[0]
|
||||||
case *api.PodList:
|
if resultValue.IsNil() {
|
||||||
h.printHeader(podColumns, w)
|
return nil
|
||||||
return h.printPodList(o, w)
|
} else {
|
||||||
case *api.ReplicationController:
|
return resultValue.Interface().(error)
|
||||||
h.printHeader(replicationControllerColumns, w)
|
}
|
||||||
return h.printReplicationController(o, w)
|
} else {
|
||||||
case *api.ReplicationControllerList:
|
return fmt.Errorf("Error: unknown type %#v", obj)
|
||||||
h.printHeader(replicationControllerColumns, w)
|
|
||||||
return h.printReplicationControllerList(o, w)
|
|
||||||
case *api.Service:
|
|
||||||
h.printHeader(serviceColumns, w)
|
|
||||||
return h.printService(o, w)
|
|
||||||
case *api.ServiceList:
|
|
||||||
h.printHeader(serviceColumns, w)
|
|
||||||
return h.printServiceList(o, w)
|
|
||||||
case *api.Minion:
|
|
||||||
h.printHeader(minionColumns, w)
|
|
||||||
return h.printMinion(o, w)
|
|
||||||
case *api.MinionList:
|
|
||||||
h.printHeader(minionColumns, w)
|
|
||||||
return h.printMinionList(o, w)
|
|
||||||
case *api.Status:
|
|
||||||
return h.printStatus(o, w)
|
|
||||||
default:
|
|
||||||
_, err := fmt.Fprintf(w, "Error: unknown type %#v", obj)
|
|
||||||
return err
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -19,6 +19,8 @@ package kubecfg
|
|||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
"reflect"
|
"reflect"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
@@ -101,3 +103,56 @@ func TestIdentityPrinter(t *testing.T) {
|
|||||||
t.Errorf("Unexpected inequality: %#v vs %#v", obj, objOut)
|
t.Errorf("Unexpected inequality: %#v vs %#v", obj, objOut)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type TestPrintType struct {
|
||||||
|
Data string
|
||||||
|
}
|
||||||
|
|
||||||
|
type TestUnknownType struct{}
|
||||||
|
|
||||||
|
func PrintCustomType(obj *TestPrintType, w io.Writer) error {
|
||||||
|
_, err := fmt.Fprintf(w, "%s", obj.Data)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func ErrorPrintHandler(obj *TestPrintType, w io.Writer) error {
|
||||||
|
return fmt.Errorf("ErrorPrintHandler error")
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestCustomTypePrinting(t *testing.T) {
|
||||||
|
columns := []string{"Data"}
|
||||||
|
printer := NewHumanReadablePrinter()
|
||||||
|
printer.Handler(columns, PrintCustomType)
|
||||||
|
|
||||||
|
obj := TestPrintType{"test object"}
|
||||||
|
buffer := &bytes.Buffer{}
|
||||||
|
err := printer.PrintObj(&obj, buffer)
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("An error occurred printing the custom type: %#v", err)
|
||||||
|
}
|
||||||
|
expectedOutput := "Data\n----------\ntest object"
|
||||||
|
if buffer.String() != expectedOutput {
|
||||||
|
t.Errorf("The data was not printed as expected. Expected:\n%s\nGot:\n%s", expectedOutput, buffer.String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPrintHandlerError(t *testing.T) {
|
||||||
|
columns := []string{"Data"}
|
||||||
|
printer := NewHumanReadablePrinter()
|
||||||
|
printer.Handler(columns, ErrorPrintHandler)
|
||||||
|
obj := TestPrintType{"test object"}
|
||||||
|
buffer := &bytes.Buffer{}
|
||||||
|
err := printer.PrintObj(&obj, buffer)
|
||||||
|
if err == nil || err.Error() != "ErrorPrintHandler error" {
|
||||||
|
t.Errorf("Did not get the expected error: %#v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestUnknownTypePrinting(t *testing.T) {
|
||||||
|
printer := NewHumanReadablePrinter()
|
||||||
|
buffer := &bytes.Buffer{}
|
||||||
|
err := printer.PrintObj(&TestUnknownType{}, buffer)
|
||||||
|
if err == nil {
|
||||||
|
t.Errorf("An error was expected from printing unknown type")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Reference in New Issue
Block a user