2
0

add_env_group.go 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  1. package stack
  2. import (
  3. "fmt"
  4. "net/http"
  5. "strings"
  6. "github.com/porter-dev/porter/api/server/authz"
  7. "github.com/porter-dev/porter/api/server/handlers"
  8. "github.com/porter-dev/porter/api/server/shared"
  9. "github.com/porter-dev/porter/api/server/shared/apierrors"
  10. "github.com/porter-dev/porter/api/server/shared/config"
  11. "github.com/porter-dev/porter/api/types"
  12. "github.com/porter-dev/porter/internal/kubernetes/envgroup"
  13. "github.com/porter-dev/porter/internal/models"
  14. "github.com/porter-dev/porter/internal/stacks"
  15. )
  16. type StackAddEnvGroupHandler struct {
  17. handlers.PorterHandlerReadWriter
  18. authz.KubernetesAgentGetter
  19. }
  20. func NewStackAddEnvGroupHandler(
  21. config *config.Config,
  22. reader shared.RequestDecoderValidator,
  23. writer shared.ResultWriter,
  24. ) *StackAddEnvGroupHandler {
  25. return &StackAddEnvGroupHandler{
  26. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, reader, writer),
  27. KubernetesAgentGetter: authz.NewOutOfClusterAgentGetter(config),
  28. }
  29. }
  30. func (p *StackAddEnvGroupHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  31. proj, _ := r.Context().Value(types.ProjectScope).(*models.Project)
  32. cluster, _ := r.Context().Value(types.ClusterScope).(*models.Cluster)
  33. namespace, _ := r.Context().Value(types.NamespaceScope).(string)
  34. stack, _ := r.Context().Value(types.StackScope).(*models.Stack)
  35. req := &types.CreateStackEnvGroupRequest{}
  36. if ok := p.DecodeAndValidate(w, r, req); !ok {
  37. return
  38. }
  39. if len(stack.Revisions) == 0 {
  40. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(
  41. fmt.Errorf("no stack revisions exist"), http.StatusBadRequest,
  42. ))
  43. return
  44. }
  45. latestRevision, err := p.Repo().Stack().ReadStackRevisionByNumber(stack.ID, stack.Revisions[0].RevisionNumber)
  46. if err != nil {
  47. p.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  48. return
  49. }
  50. newSourceConfigs, err := stacks.CloneSourceConfigs(latestRevision.SourceConfigs)
  51. if err != nil {
  52. p.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  53. return
  54. }
  55. appResources, err := stacks.CloneAppResources(latestRevision.Resources, latestRevision.SourceConfigs, newSourceConfigs)
  56. if err != nil {
  57. p.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  58. return
  59. }
  60. envGroups, err := stacks.CloneEnvGroups(latestRevision.EnvGroups)
  61. if err != nil {
  62. p.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  63. return
  64. }
  65. newEnvGroups, err := getEnvGroupModels([]*types.CreateStackEnvGroupRequest{req}, proj.ID, cluster.ID, namespace)
  66. if err != nil {
  67. p.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  68. return
  69. }
  70. envGroups = append(envGroups, newEnvGroups...)
  71. nameValidator := make(map[string]bool)
  72. for _, eg := range envGroups {
  73. if _, ok := nameValidator[eg.Name]; ok {
  74. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(fmt.Errorf("duplicate env group name: %s", eg.Name),
  75. http.StatusBadRequest))
  76. return
  77. }
  78. nameValidator[eg.Name] = true
  79. }
  80. newRevision := &models.StackRevision{
  81. StackID: stack.ID,
  82. RevisionNumber: latestRevision.RevisionNumber + 1,
  83. Status: string(types.StackRevisionStatusDeployed),
  84. SourceConfigs: newSourceConfigs,
  85. Resources: appResources,
  86. EnvGroups: envGroups,
  87. }
  88. revision, err := p.Repo().Stack().AppendNewRevision(newRevision)
  89. if err != nil {
  90. p.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  91. return
  92. }
  93. k8sAgent, err := p.GetAgent(r, cluster, "")
  94. if err != nil {
  95. p.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  96. return
  97. }
  98. envGroupDeployErrors := make([]string, 0)
  99. cm, err := envgroup.CreateEnvGroup(k8sAgent, types.ConfigMapInput{
  100. Name: req.Name,
  101. Namespace: namespace,
  102. Variables: req.Variables,
  103. SecretVariables: req.SecretVariables,
  104. })
  105. if err != nil {
  106. envGroupDeployErrors = append(envGroupDeployErrors, fmt.Sprintf("error creating env group %s", req.Name))
  107. }
  108. // add each of the linked applications to the env group
  109. for _, appName := range req.LinkedApplications {
  110. cm, err = k8sAgent.AddApplicationToVersionedConfigMap(cm, appName)
  111. if err != nil {
  112. envGroupDeployErrors = append(envGroupDeployErrors, fmt.Sprintf("error creating env group %s", req.Name))
  113. }
  114. }
  115. if len(envGroupDeployErrors) > 0 {
  116. revision.Status = string(types.StackRevisionStatusFailed)
  117. revision.Reason = "EnvGroupDeployErr"
  118. revision.Message = strings.Join(envGroupDeployErrors, " , ")
  119. } else {
  120. revision.Status = string(types.StackRevisionStatusDeployed)
  121. revision.Reason = "AddEnvGroupSuccess"
  122. revision.Message = "Env Group " + req.Name + " added successfully."
  123. }
  124. _, err = p.Repo().Stack().UpdateStackRevision(revision)
  125. if err != nil {
  126. p.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  127. return
  128. }
  129. }