delete_deployment.go 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. package environment
  2. import (
  3. "context"
  4. "errors"
  5. "fmt"
  6. "net/http"
  7. "strings"
  8. "github.com/google/go-github/v41/github"
  9. "github.com/porter-dev/porter/api/server/authz"
  10. "github.com/porter-dev/porter/api/server/handlers"
  11. "github.com/porter-dev/porter/api/server/shared"
  12. "github.com/porter-dev/porter/api/server/shared/apierrors"
  13. "github.com/porter-dev/porter/api/server/shared/config"
  14. "github.com/porter-dev/porter/api/server/shared/requestutils"
  15. "github.com/porter-dev/porter/api/types"
  16. "github.com/porter-dev/porter/internal/models"
  17. "gorm.io/gorm"
  18. )
  19. type DeleteDeploymentHandler struct {
  20. handlers.PorterHandlerReadWriter
  21. authz.KubernetesAgentGetter
  22. }
  23. func NewDeleteDeploymentHandler(
  24. config *config.Config,
  25. decoderValidator shared.RequestDecoderValidator,
  26. writer shared.ResultWriter,
  27. ) *DeleteDeploymentHandler {
  28. return &DeleteDeploymentHandler{
  29. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  30. KubernetesAgentGetter: authz.NewOutOfClusterAgentGetter(config),
  31. }
  32. }
  33. func (c *DeleteDeploymentHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  34. project, _ := r.Context().Value(types.ProjectScope).(*models.Project)
  35. cluster, _ := r.Context().Value(types.ClusterScope).(*models.Cluster)
  36. deplID, reqErr := requestutils.GetURLParamUint(r, "deployment_id")
  37. if reqErr != nil {
  38. c.HandleAPIError(w, r, apierrors.NewErrInternal(reqErr))
  39. return
  40. }
  41. // read the deployment
  42. depl, err := c.Repo().Environment().ReadDeploymentByID(project.ID, cluster.ID, deplID)
  43. if err != nil {
  44. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  45. return
  46. }
  47. // delete corresponding namespace
  48. agent, err := c.GetAgent(r, cluster, "")
  49. if err != nil {
  50. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  51. return
  52. }
  53. // make sure we don't delete default or kube-system by checking for prefix, for now
  54. if strings.Contains(depl.Namespace, "pr-") {
  55. err = agent.DeleteNamespace(depl.Namespace)
  56. if err != nil {
  57. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  58. return
  59. }
  60. }
  61. // check that the environment belongs to the project and cluster IDs
  62. env, err := c.Repo().Environment().ReadEnvironmentByID(project.ID, cluster.ID, depl.EnvironmentID)
  63. if err != nil {
  64. if errors.Is(err, gorm.ErrRecordNotFound) {
  65. c.HandleAPIError(w, r, apierrors.NewErrForbidden(fmt.Errorf("environment id not found in cluster and project")))
  66. return
  67. }
  68. c.HandleAPIError(w, r, apierrors.NewErrInternal(reqErr))
  69. return
  70. }
  71. client, err := getGithubClientFromEnvironment(c.Config(), env)
  72. if err != nil {
  73. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  74. return
  75. }
  76. // Create new deployment status to indicate deployment is ready
  77. state := "inactive"
  78. deploymentStatusRequest := github.DeploymentStatusRequest{
  79. State: &state,
  80. }
  81. _, _, err = client.Repositories.CreateDeploymentStatus(
  82. context.Background(),
  83. env.GitRepoOwner,
  84. env.GitRepoName,
  85. depl.GHDeploymentID,
  86. &deploymentStatusRequest,
  87. )
  88. if err != nil {
  89. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  90. return
  91. }
  92. depl.Status = types.DeploymentStatusInactive
  93. // update the deployment to mark it inactive
  94. depl, err = c.Repo().Environment().UpdateDeployment(depl)
  95. if err != nil {
  96. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  97. return
  98. }
  99. c.WriteResult(w, r, depl.ToDeploymentType())
  100. }