2
0

reader.go 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194
  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. queryObj := make(map[string]interface{})
  93. queryObj["items"] = []interface{}{u.Object}
  94. data, err := utils.QueryValues(queryObj, r.Queries)
  95. if err != nil {
  96. return
  97. }
  98. pkt["data"] = data
  99. on(pkt)
  100. },
  101. UpdateFunc: func(oldObj, newObj interface{}) {
  102. pkt := make(map[string]interface{})
  103. pkt["kind"] = "update"
  104. u := newObj.(*unstructured.Unstructured)
  105. queryObj := make(map[string]interface{})
  106. queryObj["items"] = []interface{}{u.Object}
  107. data, err := utils.QueryValues(queryObj, r.Queries)
  108. if err != nil {
  109. return
  110. }
  111. pkt["data"] = data
  112. on(pkt)
  113. },
  114. DeleteFunc: func(obj interface{}) {
  115. pkt := make(map[string]interface{})
  116. pkt["kind"] = "delete"
  117. u := obj.(*unstructured.Unstructured)
  118. queryObj := make(map[string]interface{})
  119. queryObj["items"] = []interface{}{u.Object}
  120. data, err := utils.QueryValues(queryObj, r.Queries)
  121. if err != nil {
  122. return
  123. }
  124. pkt["data"] = data
  125. on(pkt)
  126. },
  127. })
  128. go informer.Run(stopCh)
  129. return nil
  130. }
  131. func (r *TemplateReader) valuesFromList() (map[string]interface{}, error) {
  132. list, err := r.resource.List(context.TODO(), metav1.ListOptions{})
  133. if err != nil {
  134. return nil, err
  135. }
  136. return list.UnstructuredContent(), nil
  137. }
  138. func (r *TemplateReader) valuesFromGet() (map[string]interface{}, error) {
  139. get, err := r.resource.Get(context.TODO(), r.Object.Name, metav1.GetOptions{})
  140. if err != nil {
  141. return nil, err
  142. }
  143. return get.Object, nil
  144. }