| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385 |
- package cmd
- import (
- "context"
- "fmt"
- "io/ioutil"
- "os"
- "reflect"
- "strings"
- "text/tabwriter"
- "time"
- "k8s.io/apimachinery/pkg/runtime/schema"
- "k8s.io/client-go/dynamic"
- di "k8s.io/client-go/dynamic/dynamicinformer"
- "k8s.io/client-go/tools/cache"
- "k8s.io/client-go/tools/clientcmd"
- "k8s.io/client-go/util/jsonpath"
- "github.com/fatih/color"
- "github.com/porter-dev/porter/internal/kubernetes/local"
- "github.com/spf13/cobra"
- "gopkg.in/yaml.v2"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
- )
- type Object struct {
- Group string `yaml:"Group"`
- Version string `yaml:"Version"`
- Resource string `yaml:"Resource"`
- }
- type Field struct {
- Kind string `yaml:"Kind"`
- Query string `yaml:"Query"`
- }
- type Operation struct {
- Kind string `yaml:"Kind"`
- Object *Object `yaml:"Object"`
- Fields []*Field `yaml:"Fields"`
- }
- func getOperations() ([]*Operation, error) {
- ops := make([]*Operation, 0)
- yamlFile, err := ioutil.ReadFile("./cmd/test.yaml")
- if err != nil {
- return nil, err
- }
- err = yaml.Unmarshal(yamlFile, &ops)
- if err != nil {
- return nil, err
- }
- return ops, nil
- }
- func initTabWriterArgs(fields []*Field) (string, []interface{}) {
- firstArg := ""
- nextArgs := make([]interface{}, 0)
- for i, field := range fields {
- firstArg += "%s"
- if len(fields) > i+1 {
- firstArg += "\t"
- } else {
- firstArg += "\n"
- }
- nextArgs = append(nextArgs, strings.ToUpper(field.Kind))
- }
- return firstArg, nextArgs
- }
- func PrintList(op *Operation, items []unstructured.Unstructured) {
- w := new(tabwriter.Writer)
- w.Init(os.Stdout, 3, 8, 0, '\t', tabwriter.AlignRight)
- firstTWArg, twHeaders := initTabWriterArgs(op.Fields)
- fmt.Fprintf(w, firstTWArg, twHeaders...)
- for _, item := range items {
- printRes := make([]interface{}, 0)
- for _, field := range op.Fields {
- name := item.GetName()
- j := jsonpath.New(name)
- j.AllowMissingKeys(true)
- err := j.Parse(field.Query)
- if err != nil {
- fmt.Printf("could not parse query for object %s: error=%s", name, err)
- continue
- }
- fullResults, err := j.FindResults(item.Object)
- if err != nil {
- fmt.Printf("query error for object %s: error=%s", name, err)
- continue
- }
- res := make([]string, 0)
- for ix := range fullResults {
- for _, result := range fullResults[ix] {
- res = append(res, fmt.Sprintf("%v", reflect.ValueOf(result.Interface())))
- }
- }
- switch field.Kind {
- case "Title", "Status":
- printRes = append(printRes, strings.Join(res, ""))
- case "Array":
- printRes = append(printRes, strings.Join(res, ","))
- }
- }
- fmt.Fprintf(w, firstTWArg, printRes...)
- }
- w.Flush()
- }
- func PrintRead(op *Operation, item *unstructured.Unstructured) {
- w := new(tabwriter.Writer)
- w.Init(os.Stdout, 3, 8, 0, '\t', tabwriter.AlignRight)
- firstTWArg, twHeaders := initTabWriterArgs(op.Fields)
- fmt.Fprintf(w, firstTWArg, twHeaders...)
- printRes := make([]interface{}, 0)
- for _, field := range op.Fields {
- name := item.GetName()
- j := jsonpath.New(name)
- j.AllowMissingKeys(true)
- err := j.Parse(field.Query)
- if err != nil {
- fmt.Printf("could not parse query for object %s: error=%s", name, err)
- continue
- }
- fullResults, err := j.FindResults(item.Object)
- if err != nil {
- fmt.Printf("query error for object %s: error=%s", name, err)
- continue
- }
- res := make([]string, 0)
- for ix := range fullResults {
- for _, result := range fullResults[ix] {
- res = append(res, fmt.Sprintf("%v", reflect.ValueOf(result.Interface())))
- }
- }
- switch field.Kind {
- case "Title", "Status":
- printRes = append(printRes, strings.Join(res, ""))
- case "Array":
- printRes = append(printRes, strings.Join(res, ","))
- }
- }
- fmt.Fprintf(w, firstTWArg, printRes...)
- w.Flush()
- }
- func listOperation(client dynamic.Interface, op *Operation) {
- objRes := schema.GroupVersionResource{
- Group: op.Object.Group,
- Version: op.Object.Version,
- Resource: op.Object.Resource,
- }
- namespace := "default"
- fmt.Printf("Listing deployments in namespace %q:\n", namespace)
- list, err := client.Resource(objRes).Namespace(namespace).List(context.TODO(), metav1.ListOptions{})
- if err != nil {
- red := color.New(color.FgRed)
- red.Println("Error:", err.Error())
- os.Exit(1)
- }
- PrintList(op, list.Items)
- }
- var testCmd = &cobra.Command{
- Use: "test",
- Short: "Testing",
- Run: func(cmd *cobra.Command, args []string) {
- contextName := "gke_porter-dev-273614_us-central1-f_c-cz2vr"
- rawBytes, err := local.GetKubeconfigFromHost("", []string{contextName})
- if err != nil {
- red := color.New(color.FgRed)
- red.Println("Error:", err.Error())
- os.Exit(1)
- }
- conf, err := clientcmd.NewClientConfigFromBytes(rawBytes)
- rawConf, err := conf.RawConfig()
- conf = clientcmd.NewDefaultClientConfig(rawConf, &clientcmd.ConfigOverrides{
- CurrentContext: contextName,
- })
- restConf, err := conf.ClientConfig()
- if err != nil {
- red := color.New(color.FgRed)
- red.Println("Error:", err.Error())
- os.Exit(1)
- }
- client, err := dynamic.NewForConfig(restConf)
- if err != nil {
- red := color.New(color.FgRed)
- red.Println("Error:", err.Error())
- os.Exit(1)
- }
- ops, err := getOperations()
- if err != nil {
- red := color.New(color.FgRed)
- red.Println("Error:", err.Error())
- os.Exit(1)
- }
- StreamDynamic(client, ops[0])
- // listOperation(client, ops[0])
- },
- }
- type Message struct {
- EventType string
- Object interface{}
- Kind string
- }
- func StreamDynamic(client dynamic.Interface, op *Operation) error {
- factory := di.NewDynamicSharedInformerFactory(
- client,
- 10*time.Second,
- )
- objRes := schema.GroupVersionResource{
- Group: op.Object.Group,
- Version: op.Object.Version,
- Resource: op.Object.Resource,
- }
- informer := factory.ForResource(objRes).Informer()
- stopper := make(chan struct{})
- errorchan := make(chan error)
- defer close(errorchan)
- defer close(stopper)
- informer.AddEventHandler(cache.ResourceEventHandlerFuncs{
- UpdateFunc: func(oldObj, newObj interface{}) {
- u := newObj.(*unstructured.Unstructured)
- PrintRead(op, u)
- },
- DeleteFunc: func(obj interface{}) {
- u := obj.(*unstructured.Unstructured)
- PrintRead(op, u)
- },
- })
- // TODO -- websocket
- // go func() {
- // // listens for websocket closing handshake
- // for {
- // if _, _, err := conn.ReadMessage(); err != nil {
- // defer conn.Close()
- // defer close(stopper)
- // defer fmt.Println("Successfully closed controller status stream")
- // errorchan <- nil
- // return
- // }
- // }
- // }()
- go informer.Run(stopper)
- for {
- select {
- case err := <-errorchan:
- return err
- }
- }
- }
- func init() {
- rootCmd.AddCommand(testCmd)
- }
- // replace arrays and scalar values
- func CoalesceValues(base, override map[string]interface{}) map[string]interface{} {
- for key, val := range base {
- if oVal, ok := override[key]; ok {
- if oVal == nil {
- delete(override, key)
- } else if oMapVal, ok := oVal.(map[string]interface{}); ok {
- bMapVal, ok := val.(map[string]interface{})
- if !ok {
- continue
- }
- override[key] = mergeMaps(bMapVal, oMapVal)
- }
- } else {
- override[key] = val
- }
- }
- return override
- }
- func isYAMLTable(v interface{}) bool {
- _, ok := v.(map[string]interface{})
- return ok
- }
- // mergeMaps merges any number of maps together, with maps later in the slice taking
- // precedent
- func mergeMaps(maps ...map[string]interface{}) map[string]interface{} {
- // merge bottom-up
- if len(maps) > 2 {
- mLen := len(maps)
- newMaps := maps[0 : mLen-2]
- // reduce length of maps by 1 and merge again
- newMaps = append(newMaps, mergeMaps(maps[mLen-2], maps[mLen-1]))
- return mergeMaps(newMaps...)
- } else if len(maps) == 2 {
- if maps[0] == nil {
- return maps[1]
- }
- if maps[1] == nil {
- return maps[0]
- }
- for key, map0Val := range maps[0] {
- if map1Val, ok := maps[1][key]; ok && map1Val == nil {
- delete(maps[1], key)
- } else if !ok {
- maps[1][key] = map0Val
- } else if isYAMLTable(map0Val) {
- if isYAMLTable(map1Val) {
- mergeMaps(map0Val.(map[string]interface{}), map1Val.(map[string]interface{}))
- }
- }
- }
- return maps[1]
- } else if len(maps) == 1 {
- return maps[0]
- }
- return nil
- }
|