delete.go 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  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. err = p.Config().BillingManager.DeleteCustomer(proj)
  85. if err != nil {
  86. e := "error deleting project in billing provider"
  87. err = telemetry.Error(ctx, span, err, e)
  88. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  89. return
  90. }
  91. deletedProject, err := p.Repo().Project().DeleteProject(proj)
  92. if err != nil {
  93. e := "error deleting project"
  94. err = telemetry.Error(ctx, span, err, e)
  95. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  96. return
  97. }
  98. err = p.Repo().AWSAssumeRoleChainer().Delete(ctx, proj.ID)
  99. if err != nil {
  100. e := "error deleting assume role chain"
  101. err = telemetry.Error(ctx, span, err, e)
  102. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  103. return
  104. }
  105. err = p.Repo().Project().DeleteRolesForProject(proj.ID)
  106. if err != nil {
  107. e := "error deleting roles for project"
  108. err = telemetry.Error(ctx, span, err, e)
  109. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  110. return
  111. }
  112. p.WriteResult(w, r, deletedProject.ToProjectType(p.Config().LaunchDarklyClient))
  113. }