| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798 |
- package project
- import (
- "fmt"
- "net/http"
- "connectrpc.com/connect"
- porterv1 "github.com/porter-dev/api-contracts/generated/go/porter/v1"
- "github.com/porter-dev/porter/api/server/handlers"
- "github.com/porter-dev/porter/api/server/shared"
- "github.com/porter-dev/porter/api/server/shared/apierrors"
- "github.com/porter-dev/porter/api/server/shared/config"
- "github.com/porter-dev/porter/api/types"
- "github.com/porter-dev/porter/internal/models"
- "github.com/porter-dev/porter/internal/notifier"
- )
- type ProjectDeleteHandler struct {
- handlers.PorterHandlerWriter
- }
- func NewProjectDeleteHandler(
- config *config.Config,
- writer shared.ResultWriter,
- ) *ProjectDeleteHandler {
- return &ProjectDeleteHandler{
- PorterHandlerWriter: handlers.NewDefaultPorterHandler(config, nil, writer),
- }
- }
- func (p *ProjectDeleteHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
- ctx := r.Context()
- user, _ := ctx.Value(types.UserScope).(*models.User)
- proj, _ := ctx.Value(types.ProjectScope).(*models.Project)
- if proj.CapiProvisionerEnabled {
- clusters, err := p.Config().Repo.Cluster().ListClustersByProjectID(proj.ID)
- if err != nil {
- p.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error finding clusters for project: %w", err)))
- return
- }
- for _, cluster := range clusters {
- if cluster.ProvisionedBy == "CAPI" {
- if !cluster.DeletedAt.Time.IsZero() {
- continue
- }
- contractRevision, err := p.Config().Repo.APIContractRevisioner().List(ctx, proj.ID, cluster.ID)
- if err != nil {
- p.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error finding contract revisions for cluster: %w", err)))
- return
- }
- if len(contractRevision) == 0 {
- continue
- }
- req := connect.NewRequest(&porterv1.DeleteClusterRequest{
- ContractRevision: &porterv1.ContractRevision{
- ClusterId: int32(cluster.ID),
- ProjectId: int32(cluster.ProjectID),
- RevisionId: contractRevision[0].ID.String(),
- },
- })
- _, err = p.Config().ClusterControlPlaneClient.DeleteCluster(ctx, req)
- if err != nil {
- p.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error deleting cluster: %w", err)))
- return
- }
- }
- }
- }
- err := p.Config().UserNotifier.SendProjectDeleteEmail(
- ¬ifier.SendProjectDeleteEmailOpts{
- Email: user.Email,
- Project: proj.Name,
- },
- )
- if err != nil {
- p.HandleAPIError(w, r, apierrors.NewErrInternal(err))
- return
- }
- deletedProject, err := p.Repo().Project().DeleteProject(proj)
- if err != nil {
- p.HandleAPIError(w, r, apierrors.NewErrInternal(err))
- return
- }
- p.WriteResult(w, r, deletedProject.ToProjectType(p.Config().LaunchDarklyClient))
- // delete the billing team
- if err := p.Config().BillingManager.DeleteTeam(user, proj); err != nil {
- // we do not write error response, since setting up billing error can be
- // resolved later and may not be fatal
- p.HandleAPIErrorNoWrite(w, r, apierrors.NewErrInternal(err))
- }
- }
|