delete.go 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  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/telemetry"
  14. )
  15. type ProjectDeleteHandler struct {
  16. handlers.PorterHandlerWriter
  17. }
  18. func NewProjectDeleteHandler(
  19. config *config.Config,
  20. writer shared.ResultWriter,
  21. ) *ProjectDeleteHandler {
  22. return &ProjectDeleteHandler{
  23. PorterHandlerWriter: handlers.NewDefaultPorterHandler(config, nil, writer),
  24. }
  25. }
  26. func (p *ProjectDeleteHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  27. ctx, span := telemetry.NewSpan(r.Context(), "delete-project")
  28. defer span.End()
  29. user, _ := ctx.Value(types.UserScope).(*models.User)
  30. proj, _ := ctx.Value(types.ProjectScope).(*models.Project)
  31. if proj.GetFeatureFlag(models.CapiProvisionerEnabled, p.Config().LaunchDarklyClient) {
  32. clusters, err := p.Config().Repo.Cluster().ListClustersByProjectID(proj.ID)
  33. if err != nil {
  34. e := "error finding clusters for project"
  35. err = telemetry.Error(ctx, span, err, e)
  36. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  37. return
  38. }
  39. for _, cluster := range clusters {
  40. if cluster.ProvisionedBy == "CAPI" {
  41. if !cluster.DeletedAt.Time.IsZero() {
  42. continue
  43. }
  44. contractRevision, err := p.Config().Repo.APIContractRevisioner().List(ctx, proj.ID, cluster.ID)
  45. if err != nil {
  46. e := "error finding contract revisions for cluster"
  47. err = telemetry.Error(ctx, span, err, e)
  48. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  49. return
  50. }
  51. if len(contractRevision) == 0 {
  52. continue
  53. }
  54. req := connect.NewRequest(&porterv1.DeleteClusterRequest{
  55. ContractRevision: &porterv1.ContractRevision{
  56. ClusterId: int32(cluster.ID),
  57. ProjectId: int32(cluster.ProjectID),
  58. RevisionId: contractRevision[0].ID.String(),
  59. },
  60. })
  61. _, err = p.Config().ClusterControlPlaneClient.DeleteCluster(ctx, req)
  62. if err != nil {
  63. e := "error deleting cluster"
  64. err = telemetry.Error(ctx, span, err, e)
  65. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  66. return
  67. }
  68. }
  69. }
  70. }
  71. err := p.Config().UserNotifier.SendProjectDeleteEmail(
  72. &notifier.SendProjectDeleteEmailOpts{
  73. Email: user.Email,
  74. Project: proj.Name,
  75. },
  76. )
  77. if err != nil {
  78. e := "error sending project deletion email"
  79. err = telemetry.Error(ctx, span, err, e)
  80. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  81. return
  82. }
  83. deletedProject, err := p.Repo().Project().DeleteProject(proj)
  84. if err != nil {
  85. e := "error deleting project"
  86. err = telemetry.Error(ctx, span, err, e)
  87. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  88. return
  89. }
  90. err = p.Repo().AWSAssumeRoleChainer().Delete(ctx, proj.ID)
  91. if err != nil {
  92. e := "error deleting assume role chain"
  93. err = telemetry.Error(ctx, span, err, e)
  94. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  95. return
  96. }
  97. err = p.Repo().Project().DeleteRolesForProject(proj.ID)
  98. if err != nil {
  99. e := "error deleting roles for project"
  100. err = telemetry.Error(ctx, span, err, e)
  101. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  102. return
  103. }
  104. p.WriteResult(w, r, deletedProject.ToProjectType(p.Config().LaunchDarklyClient))
  105. // delete the billing team
  106. if err := p.Config().BillingManager.DeleteTeam(user, proj); err != nil {
  107. // we do not write error response, since setting up billing error can be
  108. // resolved later and may not be fatal
  109. p.HandleAPIErrorNoWrite(w, r, apierrors.NewErrInternal(err))
  110. }
  111. }