parser.go 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  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. DynamicObject *td.Object
  20. HelmAgent *helm.Agent
  21. HelmRelease *release.Release
  22. HelmChart *chart.Chart
  23. }
  24. func FormYAMLFromBytes(def *ClientConfigDefault, bytes []byte) (*models.FormYAML, error) {
  25. form, err := unqueriedFormYAMLFromBytes(bytes)
  26. if err != nil {
  27. return nil, err
  28. }
  29. lookup := formToLookupTable(def, form)
  30. // merge data from lookup
  31. data := make(map[string]interface{})
  32. for _, lookupVal := range lookup {
  33. queryRes, err := lookupVal.TemplateReader.Read()
  34. if err != nil {
  35. continue
  36. }
  37. for queryResKey, queryResVal := range queryRes {
  38. data[queryResKey] = queryResVal
  39. }
  40. }
  41. for i, tab := range form.Tabs {
  42. for j, section := range tab.Sections {
  43. for k, content := range section.Contents {
  44. key := fmt.Sprintf("tabs[%d].sections[%d].contents[%d]", i, j, k)
  45. if val, ok := data[key]; ok {
  46. content.Value = val
  47. }
  48. }
  49. }
  50. }
  51. return form, nil
  52. }
  53. // unqueriedFormYAMLFromBytes returns a FormYAML without values queries populated
  54. func unqueriedFormYAMLFromBytes(bytes []byte) (*models.FormYAML, error) {
  55. // parse bytes into object
  56. form := &models.FormYAML{}
  57. err := yaml.Unmarshal(bytes, form)
  58. if err != nil {
  59. return nil, err
  60. }
  61. // populate all context fields, with default set to helm/values with no config
  62. parent := &models.FormContext{
  63. Type: "helm/values",
  64. }
  65. for _, tab := range form.Tabs {
  66. if tab.Context == nil {
  67. tab.Context = parent
  68. }
  69. for _, section := range tab.Sections {
  70. if section.Context == nil {
  71. section.Context = tab.Context
  72. }
  73. for _, content := range section.Contents {
  74. if content.Context == nil {
  75. content.Context = section.Context
  76. }
  77. }
  78. }
  79. }
  80. return form, nil
  81. }
  82. type ContextConfig struct {
  83. FromType string // "live" or "declared"
  84. Capabilities []string // "read", "write"
  85. TemplateReader templater.TemplateReader
  86. TemplateWriter templater.TemplateWriter
  87. }
  88. // create map[*FormContext]*ContextConfig
  89. // assumes all contexts populated
  90. func formToLookupTable(def *ClientConfigDefault, form *models.FormYAML) map[*models.FormContext]*ContextConfig {
  91. lookup := make(map[*models.FormContext]*ContextConfig)
  92. for i, tab := range form.Tabs {
  93. for j, section := range tab.Sections {
  94. for k, content := range section.Contents {
  95. if content.Context == nil {
  96. continue
  97. }
  98. if _, ok := lookup[content.Context]; !ok {
  99. lookup[content.Context] = formContextToContextConfig(def, content.Context)
  100. }
  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. }
  112. }
  113. }
  114. return lookup
  115. }
  116. // TODO -- this needs to be able to construct new context configs based on
  117. // configuration for each context, but right now just uses the default config
  118. // for everything
  119. func formContextToContextConfig(def *ClientConfigDefault, context *models.FormContext) *ContextConfig {
  120. res := &ContextConfig{}
  121. switch context.Type {
  122. case "helm/values":
  123. res.FromType = "declared"
  124. res.Capabilities = []string{"read", "write"}
  125. res.TemplateReader = &th.ValuesTemplateReader{
  126. Release: def.HelmRelease,
  127. Chart: def.HelmChart,
  128. }
  129. relName := ""
  130. if def.HelmRelease != nil {
  131. relName = def.HelmRelease.Name
  132. }
  133. res.TemplateWriter = &th.ValuesTemplateWriter{
  134. Agent: def.HelmAgent,
  135. Chart: def.HelmChart,
  136. ReleaseName: relName,
  137. }
  138. case "helm/manifests":
  139. res.FromType = "live"
  140. res.Capabilities = []string{"read"}
  141. res.TemplateReader = &th.ManifestsTemplateReader{
  142. Release: def.HelmRelease,
  143. }
  144. case "cluster":
  145. res.FromType = "live"
  146. res.Capabilities = []string{"read"}
  147. res.TemplateReader = td.NewDynamicTemplateReader(def.DynamicClient, def.DynamicObject)
  148. default:
  149. return nil
  150. }
  151. return res
  152. }