2
0

utils.go 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. package preview
  2. import (
  3. "context"
  4. "fmt"
  5. "os"
  6. "strconv"
  7. "strings"
  8. "github.com/porter-dev/porter/api/types"
  9. "github.com/porter-dev/porter/cli/cmd/config"
  10. "github.com/porter-dev/porter/internal/integrations/preview"
  11. )
  12. func GetSource(projectID uint, resourceName string, input map[string]interface{}) (*preview.Source, error) {
  13. output := &preview.Source{}
  14. // first read from env vars
  15. output.Name = os.Getenv("PORTER_SOURCE_NAME")
  16. output.Repo = os.Getenv("PORTER_SOURCE_REPO")
  17. output.Version = os.Getenv("PORTER_SOURCE_VERSION")
  18. // next, check for values in the YAML file
  19. if output.Name == "" {
  20. if name, ok := input["name"]; ok {
  21. nameVal, ok := name.(string)
  22. if !ok {
  23. return nil, fmt.Errorf("error parsing source for resource '%s': invalid name provided", resourceName)
  24. }
  25. output.Name = nameVal
  26. }
  27. }
  28. if output.Name == "" {
  29. return nil, fmt.Errorf("error parsing source for resource '%s': source name required", resourceName)
  30. }
  31. if output.Repo == "" {
  32. if repo, ok := input["repo"]; ok {
  33. repoVal, ok := repo.(string)
  34. if !ok {
  35. return nil, fmt.Errorf("error parsing source for resource '%s': invalid repo provided", resourceName)
  36. }
  37. output.Repo = repoVal
  38. }
  39. }
  40. if output.Version == "" {
  41. if version, ok := input["version"]; ok {
  42. versionVal, ok := version.(string)
  43. if !ok {
  44. return nil, fmt.Errorf("error parsing source for resource '%s': invalid version provided", resourceName)
  45. }
  46. output.Version = versionVal
  47. }
  48. }
  49. // lastly, just put in the defaults
  50. if output.Version == "" {
  51. output.Version = "latest"
  52. }
  53. apiClient := config.GetAPIClient()
  54. serverMetadata, err := apiClient.GetPorterInstanceMetadata(context.Background())
  55. if err != nil {
  56. return nil, fmt.Errorf("error fetching Porter instance metadata: %w", err)
  57. }
  58. if output.Repo == "" {
  59. if serverMetadata.DefaultAppHelmRepoURL != "" {
  60. output.Repo = serverMetadata.DefaultAppHelmRepoURL
  61. } else {
  62. output.Repo = "https://charts.getporter.dev"
  63. }
  64. values, err := existsInRepo(projectID, output.Name, output.Version, output.Repo)
  65. if err == nil {
  66. output.SourceValues = values
  67. output.IsApplication = true
  68. return output, nil
  69. }
  70. if serverMetadata.DefaultAddonHelmRepoURL != "" {
  71. output.Repo = serverMetadata.DefaultAddonHelmRepoURL
  72. } else {
  73. output.Repo = "https://chart-addons.getporter.dev"
  74. }
  75. values, err = existsInRepo(projectID, output.Name, output.Version, output.Repo)
  76. if err == nil {
  77. output.SourceValues = values
  78. return output, nil
  79. }
  80. return nil, fmt.Errorf("error parsing source for resource '%s': source chart does not exist in the default "+
  81. "Helm repositories", resourceName)
  82. } else {
  83. // we look in the passed-in repo
  84. values, err := existsInRepo(projectID, output.Name, output.Version, output.Repo)
  85. if err == nil {
  86. output.SourceValues = values
  87. output.IsApplication = output.Repo == serverMetadata.DefaultAppHelmRepoURL || output.Repo == "https://charts.getporter.dev"
  88. return output, nil
  89. }
  90. }
  91. return nil, fmt.Errorf("error parsing source for resource '%s': source '%s' does not exist in repo '%s'",
  92. resourceName, output.Name, output.Repo)
  93. }
  94. func GetTarget(resourceName string, input map[string]interface{}) (*preview.Target, error) {
  95. output := &preview.Target{}
  96. // first read from env vars
  97. if projectEnv := os.Getenv("PORTER_PROJECT"); projectEnv != "" {
  98. project, err := strconv.Atoi(projectEnv)
  99. if err != nil {
  100. return nil, fmt.Errorf("error parsing target for resource '%s': %w", resourceName, err)
  101. }
  102. output.Project = uint(project)
  103. }
  104. if clusterEnv := os.Getenv("PORTER_CLUSTER"); clusterEnv != "" {
  105. cluster, err := strconv.Atoi(clusterEnv)
  106. if err != nil {
  107. return nil, fmt.Errorf("error parsing target for resource '%s': %w", resourceName, err)
  108. }
  109. output.Cluster = uint(cluster)
  110. }
  111. output.Namespace = getNamespace()
  112. // next, check for values in the YAML file
  113. if output.Project == 0 {
  114. if project, ok := input["project"]; ok {
  115. projectVal, ok := project.(uint)
  116. if !ok {
  117. return nil, fmt.Errorf("error parsing target for resource '%s': project value must be an integer", resourceName)
  118. }
  119. output.Project = projectVal
  120. }
  121. }
  122. if output.Cluster == 0 {
  123. if cluster, ok := input["cluster"]; ok {
  124. clusterVal, ok := cluster.(uint)
  125. if !ok {
  126. return nil, fmt.Errorf("error parsing target for resource '%s': cluster value must be an integer",
  127. resourceName)
  128. }
  129. output.Cluster = clusterVal
  130. }
  131. }
  132. if output.Namespace == "" {
  133. if namespace, ok := input["namespace"]; ok {
  134. namespaceVal, ok := namespace.(string)
  135. if !ok {
  136. return nil, fmt.Errorf("error parsing target for resource '%s': invalid namespace provided", resourceName)
  137. }
  138. output.Namespace = namespaceVal
  139. }
  140. }
  141. if appName, ok := input["app_name"]; ok {
  142. appNameVal, ok := appName.(string)
  143. if !ok {
  144. return nil, fmt.Errorf("error parsing target for resource '%s': invalid app_name provided", resourceName)
  145. }
  146. output.AppName = appNameVal
  147. }
  148. // lastly, just put in the defaults
  149. if output.Project == 0 {
  150. output.Project = config.GetCLIConfig().Project
  151. }
  152. if output.Cluster == 0 {
  153. output.Cluster = config.GetCLIConfig().Cluster
  154. }
  155. if output.Namespace == "" {
  156. output.Namespace = "default"
  157. }
  158. return output, nil
  159. }
  160. func GetNamespaceForBranchDeploy(branch, owner, name string) string {
  161. namespace := fmt.Sprintf("previewbranch-%s-%s-%s", branch,
  162. strings.ReplaceAll(strings.ToLower(owner), "_", "-"),
  163. strings.ReplaceAll(strings.ToLower(name), "_", "-"))
  164. if len(namespace) > 63 {
  165. namespace = namespace[:63] // Kubernetes' DNS 1123 label requirement
  166. }
  167. return namespace
  168. }
  169. func getNamespace() string {
  170. if owner, ok := os.LookupEnv("PORTER_REPO_OWNER"); ok {
  171. if repo, ok := os.LookupEnv("PORTER_REPO_NAME"); ok {
  172. if branchFrom, ok := os.LookupEnv("PORTER_BRANCH_FROM"); ok {
  173. if branchInto, ok := os.LookupEnv("PORTER_BRANCH_INTO"); ok {
  174. if branchInto == branchFrom { // branch deploy
  175. return GetNamespaceForBranchDeploy(branchInto, owner, repo)
  176. }
  177. }
  178. }
  179. }
  180. }
  181. return os.Getenv("PORTER_NAMESPACE")
  182. }
  183. func existsInRepo(projectID uint, name, version, url string) (map[string]interface{}, error) {
  184. chart, err := config.GetAPIClient().GetTemplate(
  185. context.Background(),
  186. projectID,
  187. name, version,
  188. &types.GetTemplateRequest{
  189. TemplateGetBaseRequest: types.TemplateGetBaseRequest{
  190. RepoURL: url,
  191. },
  192. },
  193. )
  194. if err != nil {
  195. return nil, err
  196. }
  197. return chart.Values, nil
  198. }