delete.go 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. package project
  2. import (
  3. "net/http"
  4. "connectrpc.com/connect"
  5. porterv1 "github.com/porter-dev/api-contracts/generated/go/porter/v1"
  6. "github.com/porter-dev/porter/api/server/handlers"
  7. "github.com/porter-dev/porter/api/server/shared"
  8. "github.com/porter-dev/porter/api/server/shared/apierrors"
  9. "github.com/porter-dev/porter/api/server/shared/config"
  10. "github.com/porter-dev/porter/api/types"
  11. "github.com/porter-dev/porter/internal/models"
  12. "github.com/porter-dev/porter/internal/notifier"
  13. "github.com/porter-dev/porter/internal/repository"
  14. "github.com/porter-dev/porter/internal/telemetry"
  15. )
  16. type ProjectDeleteHandler struct {
  17. handlers.PorterHandlerWriter
  18. }
  19. func NewProjectDeleteHandler(
  20. config *config.Config,
  21. writer shared.ResultWriter,
  22. ) *ProjectDeleteHandler {
  23. return &ProjectDeleteHandler{
  24. PorterHandlerWriter: handlers.NewDefaultPorterHandler(config, nil, writer),
  25. }
  26. }
  27. func (p *ProjectDeleteHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  28. ctx, span := telemetry.NewSpan(r.Context(), "delete-project")
  29. defer span.End()
  30. user, _ := ctx.Value(types.UserScope).(*models.User)
  31. proj, _ := ctx.Value(types.ProjectScope).(*models.Project)
  32. if proj.GetFeatureFlag(models.CapiProvisionerEnabled, p.Config().LaunchDarklyClient) {
  33. clusters, err := p.Config().Repo.Cluster().ListClustersByProjectID(proj.ID)
  34. if err != nil {
  35. e := "error finding clusters for project"
  36. err = telemetry.Error(ctx, span, err, e)
  37. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  38. return
  39. }
  40. for _, cluster := range clusters {
  41. if cluster.ProvisionedBy == "CAPI" {
  42. if !cluster.DeletedAt.Time.IsZero() {
  43. continue
  44. }
  45. contractRevision, err := p.Config().Repo.APIContractRevisioner().List(ctx, proj.ID, repository.WithClusterID(cluster.ID))
  46. if err != nil {
  47. e := "error finding contract revisions for cluster"
  48. err = telemetry.Error(ctx, span, err, e)
  49. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  50. return
  51. }
  52. if len(contractRevision) == 0 {
  53. continue
  54. }
  55. req := connect.NewRequest(&porterv1.DeleteClusterRequest{
  56. ContractRevision: &porterv1.ContractRevision{
  57. ClusterId: int32(cluster.ID),
  58. ProjectId: int32(cluster.ProjectID),
  59. RevisionId: contractRevision[0].ID.String(),
  60. },
  61. })
  62. _, err = p.Config().ClusterControlPlaneClient.DeleteCluster(ctx, req)
  63. if err != nil {
  64. e := "error deleting cluster"
  65. err = telemetry.Error(ctx, span, err, e)
  66. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  67. return
  68. }
  69. }
  70. }
  71. }
  72. err := p.Config().UserNotifier.SendProjectDeleteEmail(
  73. &notifier.SendProjectDeleteEmailOpts{
  74. Email: user.Email,
  75. Project: proj.Name,
  76. },
  77. )
  78. if err != nil {
  79. e := "error sending project deletion email"
  80. err = telemetry.Error(ctx, span, err, e)
  81. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  82. return
  83. }
  84. if p.Config().ServerConf.MetronomeAPIKey != "" && p.Config().ServerConf.PorterCloudPlanID != "" &&
  85. proj.GetFeatureFlag(models.MetronomeEnabled, p.Config().LaunchDarklyClient) {
  86. err = p.Config().BillingManager.MetronomeClient.EndCustomerPlan(ctx, proj.UsageID, proj.UsagePlanID)
  87. if err != nil {
  88. e := "error ending billing plan"
  89. err = telemetry.Error(ctx, span, err, e)
  90. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  91. return
  92. }
  93. telemetry.WithAttributes(span,
  94. telemetry.AttributeKV{Key: "project-id", Value: proj.ID},
  95. telemetry.AttributeKV{Key: "usage-id", Value: proj.UsageID},
  96. telemetry.AttributeKV{Key: "usage-plan-id", Value: proj.UsagePlanID},
  97. )
  98. }
  99. deletedProject, err := p.Repo().Project().DeleteProject(proj)
  100. if err != nil {
  101. e := "error deleting project"
  102. err = telemetry.Error(ctx, span, err, e)
  103. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  104. return
  105. }
  106. err = p.Repo().AWSAssumeRoleChainer().Delete(ctx, proj.ID)
  107. if err != nil {
  108. e := "error deleting assume role chain"
  109. err = telemetry.Error(ctx, span, err, e)
  110. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  111. return
  112. }
  113. err = p.Repo().Project().DeleteRolesForProject(proj.ID)
  114. if err != nil {
  115. e := "error deleting roles for project"
  116. err = telemetry.Error(ctx, span, err, e)
  117. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  118. return
  119. }
  120. p.WriteResult(w, r, deletedProject.ToProjectType(p.Config().LaunchDarklyClient))
  121. }