reader.go 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. package dynamic
  2. import (
  3. "context"
  4. "time"
  5. "github.com/porter-dev/porter/internal/templater/utils"
  6. "github.com/porter-dev/porter/internal/templater"
  7. "k8s.io/client-go/dynamic"
  8. di "k8s.io/client-go/dynamic/dynamicinformer"
  9. "k8s.io/client-go/tools/cache"
  10. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  11. "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
  12. "k8s.io/apimachinery/pkg/runtime/schema"
  13. )
  14. // Object identifies a set of k8s objects, during Group-Version-Kind, and optionally
  15. // a namespace and name to isolate a single object
  16. type Object struct {
  17. Group string
  18. Version string
  19. Resource string
  20. // Optional, if resource is namespacable
  21. Namespace string
  22. // Optional, if attempting to get an object by name
  23. Name string
  24. }
  25. // TemplateReader reads any resource registered with the k8s apiserver
  26. type TemplateReader struct {
  27. // The object to read from, identified by its group-version-kind
  28. Object *Object
  29. // The set of queries to execute, identified by a key and query string
  30. Queries []*templater.TemplateReaderQuery
  31. Client dynamic.Interface
  32. // The resource that's being queried
  33. gvr schema.GroupVersionResource
  34. resource dynamic.ResourceInterface
  35. }
  36. // NewDynamicTemplateReader creates a new DynamicTemplateReader
  37. func NewDynamicTemplateReader(client dynamic.Interface, obj *Object) templater.TemplateReader {
  38. r := &TemplateReader{
  39. Object: obj,
  40. Client: client,
  41. }
  42. objRes := schema.GroupVersionResource{
  43. Group: r.Object.Group,
  44. Version: r.Object.Version,
  45. Resource: r.Object.Resource,
  46. }
  47. // just case on the "core" group and unset it
  48. if r.Object.Group == "core" {
  49. objRes.Group = ""
  50. }
  51. r.gvr = objRes
  52. r.resource = r.Client.Resource(objRes).Namespace(r.Object.Namespace)
  53. return r
  54. }
  55. // ValuesFromTarget retrieves cluster values from the k8s apiserver
  56. func (r *TemplateReader) ValuesFromTarget() (map[string]interface{}, error) {
  57. // if name is not empty, this is a get operation
  58. if r.Object.Name != "" {
  59. return r.valuesFromGet()
  60. }
  61. return r.valuesFromList()
  62. }
  63. // RegisterQuery adds a query to the list of queries to execute
  64. func (r *TemplateReader) RegisterQuery(query *templater.TemplateReaderQuery) error {
  65. r.Queries = append(r.Queries, query)
  66. return nil
  67. }
  68. // Read returns the resulting queried data
  69. func (r *TemplateReader) Read() (map[string]interface{}, error) {
  70. values, err := r.ValuesFromTarget()
  71. if err != nil {
  72. return nil, err
  73. }
  74. return utils.QueryValues(values, r.Queries)
  75. }
  76. // ReadStream listens for CRUD operations on resources and returns resulting
  77. // queried data
  78. func (r *TemplateReader) ReadStream(
  79. on templater.OnDataStream,
  80. stopCh <-chan struct{},
  81. ) error {
  82. factory := di.NewDynamicSharedInformerFactory(
  83. r.Client,
  84. 10*time.Second,
  85. )
  86. informer := factory.ForResource(r.gvr).Informer()
  87. informer.AddEventHandler(cache.ResourceEventHandlerFuncs{
  88. AddFunc: func(obj interface{}) {
  89. pkt := make(map[string]interface{})
  90. pkt["kind"] = "create"
  91. u := obj.(*unstructured.Unstructured)
  92. data, err := utils.QueryValues(u.Object, r.Queries)
  93. if err != nil {
  94. return
  95. }
  96. pkt["data"] = data
  97. on(pkt)
  98. },
  99. UpdateFunc: func(oldObj, newObj interface{}) {
  100. pkt := make(map[string]interface{})
  101. pkt["kind"] = "update"
  102. u := newObj.(*unstructured.Unstructured)
  103. data, err := utils.QueryValues(u.Object, r.Queries)
  104. if err != nil {
  105. return
  106. }
  107. pkt["data"] = data
  108. on(pkt)
  109. },
  110. DeleteFunc: func(obj interface{}) {
  111. pkt := make(map[string]interface{})
  112. pkt["kind"] = "delete"
  113. u := obj.(*unstructured.Unstructured)
  114. data, err := utils.QueryValues(u.Object, r.Queries)
  115. if err != nil {
  116. return
  117. }
  118. pkt["data"] = data
  119. on(pkt)
  120. },
  121. })
  122. go informer.Run(stopCh)
  123. return nil
  124. }
  125. func (r *TemplateReader) valuesFromList() (map[string]interface{}, error) {
  126. list, err := r.resource.List(context.TODO(), metav1.ListOptions{})
  127. if err != nil {
  128. return nil, err
  129. }
  130. return list.UnstructuredContent(), nil
  131. }
  132. func (r *TemplateReader) valuesFromGet() (map[string]interface{}, error) {
  133. get, err := r.resource.Get(context.TODO(), r.Object.Name, metav1.GetOptions{})
  134. if err != nil {
  135. return nil, err
  136. }
  137. return get.Object, nil
  138. }