parser.go 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  1. package parser
  2. import (
  3. "fmt"
  4. "github.com/porter-dev/porter/internal/helm"
  5. "github.com/porter-dev/porter/internal/models"
  6. "github.com/porter-dev/porter/internal/templater"
  7. "github.com/porter-dev/porter/internal/templater/utils"
  8. "helm.sh/helm/v3/pkg/chart"
  9. "helm.sh/helm/v3/pkg/release"
  10. "k8s.io/client-go/dynamic"
  11. "sigs.k8s.io/yaml"
  12. td "github.com/porter-dev/porter/internal/templater/dynamic"
  13. th "github.com/porter-dev/porter/internal/templater/helm"
  14. )
  15. // TODO -- handle all continue statements, errors should at least be logged if not
  16. // thrown
  17. type ClientConfigDefault struct {
  18. DynamicClient dynamic.Interface
  19. HelmAgent *helm.Agent
  20. HelmRelease *release.Release
  21. HelmChart *chart.Chart
  22. }
  23. func FormYAMLFromBytes(def *ClientConfigDefault, bytes []byte) (*models.FormYAML, error) {
  24. form, err := unqueriedFormYAMLFromBytes(bytes)
  25. if err != nil {
  26. return nil, err
  27. }
  28. lookup := formToLookupTable(def, form)
  29. // merge data from lookup
  30. data := make(map[string]interface{})
  31. for _, lookupVal := range lookup {
  32. queryRes, err := lookupVal.TemplateReader.Read()
  33. if err != nil {
  34. continue
  35. }
  36. for queryResKey, queryResVal := range queryRes {
  37. data[queryResKey] = queryResVal
  38. }
  39. }
  40. for i, tab := range form.Tabs {
  41. for j, section := range tab.Sections {
  42. for k, content := range section.Contents {
  43. key := fmt.Sprintf("tabs[%d].sections[%d].contents[%d]", i, j, k)
  44. if val, ok := data[key]; ok {
  45. content.Value = val
  46. }
  47. }
  48. }
  49. }
  50. return form, nil
  51. }
  52. // unqueriedFormYAMLFromBytes returns a FormYAML without values queries populated
  53. func unqueriedFormYAMLFromBytes(bytes []byte) (*models.FormYAML, error) {
  54. // parse bytes into object
  55. form := &models.FormYAML{}
  56. err := yaml.Unmarshal(bytes, form)
  57. if err != nil {
  58. return nil, err
  59. }
  60. // populate all context fields, with default set to helm/values with no config
  61. parent := &models.FormContext{
  62. Type: "helm/values",
  63. }
  64. for _, tab := range form.Tabs {
  65. if tab.Context == nil {
  66. tab.Context = parent
  67. }
  68. for _, section := range tab.Sections {
  69. if section.Context == nil {
  70. section.Context = tab.Context
  71. }
  72. for _, content := range section.Contents {
  73. if content.Context == nil {
  74. content.Context = section.Context
  75. }
  76. }
  77. }
  78. }
  79. return form, nil
  80. }
  81. type ContextConfig struct {
  82. FromType string // "live" or "declared"
  83. Capabilities []string // "read", "write"
  84. TemplateReader templater.TemplateReader
  85. TemplateWriter templater.TemplateWriter
  86. }
  87. // create map[*FormContext]*ContextConfig
  88. // assumes all contexts populated
  89. func formToLookupTable(def *ClientConfigDefault, form *models.FormYAML) map[*models.FormContext]*ContextConfig {
  90. lookup := make(map[*models.FormContext]*ContextConfig)
  91. for i, tab := range form.Tabs {
  92. for j, section := range tab.Sections {
  93. for k, content := range section.Contents {
  94. if content.Context == nil {
  95. continue
  96. }
  97. if _, ok := lookup[content.Context]; !ok {
  98. lookup[content.Context] = formContextToContextConfig(def, content.Context)
  99. }
  100. if content.Value != "" {
  101. // TODO -- case on whether value is proper query string, if not resolve it to a
  102. // proper query string
  103. query, err := utils.NewQuery(
  104. fmt.Sprintf("tabs[%d].sections[%d].contents[%d]", i, j, k),
  105. fmt.Sprintf("%v", content.Value),
  106. )
  107. if err != nil {
  108. continue
  109. }
  110. lookup[content.Context].TemplateReader.RegisterQuery(query)
  111. } else if content.Variable != "" {
  112. // if variable field set without value field set, make variable field into jsonpath
  113. // query
  114. query, err := utils.NewQuery(
  115. fmt.Sprintf("tabs[%d].sections[%d].contents[%d]", i, j, k),
  116. fmt.Sprintf("{ .%v }", content.Variable),
  117. )
  118. if err != nil {
  119. continue
  120. }
  121. lookup[content.Context].TemplateReader.RegisterQuery(query)
  122. }
  123. }
  124. }
  125. }
  126. return lookup
  127. }
  128. // TODO -- this needs to be able to construct new context configs based on
  129. // configuration for each context, but right now just uses the default config
  130. // for everything
  131. func formContextToContextConfig(def *ClientConfigDefault, context *models.FormContext) *ContextConfig {
  132. res := &ContextConfig{}
  133. switch context.Type {
  134. case "helm/values":
  135. res.FromType = "declared"
  136. res.Capabilities = []string{"read", "write"}
  137. res.TemplateReader = &th.ValuesTemplateReader{
  138. Release: def.HelmRelease,
  139. Chart: def.HelmChart,
  140. }
  141. relName := ""
  142. if def.HelmRelease != nil {
  143. relName = def.HelmRelease.Name
  144. }
  145. res.TemplateWriter = &th.ValuesTemplateWriter{
  146. Agent: def.HelmAgent,
  147. Chart: def.HelmChart,
  148. ReleaseName: relName,
  149. }
  150. case "helm/manifests":
  151. res.FromType = "live"
  152. res.Capabilities = []string{"read"}
  153. res.TemplateReader = &th.ManifestsTemplateReader{
  154. Release: def.HelmRelease,
  155. }
  156. case "cluster":
  157. res.FromType = "live"
  158. res.Capabilities = []string{"read"}
  159. // identify object based on passed config
  160. obj := &td.Object{
  161. Group: context.Config["Group"],
  162. Version: context.Config["Version"],
  163. Resource: context.Config["Resource"],
  164. Namespace: context.Config["Namespace"],
  165. Name: context.Config["Name"],
  166. }
  167. res.TemplateReader = td.NewDynamicTemplateReader(def.DynamicClient, obj)
  168. default:
  169. return nil
  170. }
  171. return res
  172. }