delete.go 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139
  1. package datastore
  2. import (
  3. "context"
  4. "net/http"
  5. "connectrpc.com/connect"
  6. "github.com/google/uuid"
  7. porterv1 "github.com/porter-dev/api-contracts/generated/go/porter/v1"
  8. "github.com/porter-dev/porter/api/server/authz"
  9. "github.com/porter-dev/porter/api/server/handlers"
  10. "github.com/porter-dev/porter/api/server/handlers/release"
  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. "github.com/porter-dev/porter/internal/telemetry"
  18. )
  19. // DeleteDatastoreHandler is a struct for handling datastore deletion requests
  20. type DeleteDatastoreHandler struct {
  21. handlers.PorterHandlerReadWriter
  22. authz.KubernetesAgentGetter
  23. }
  24. // NewDeleteDatastoreHandler constructs a datastore DeleteDatastoreHandler
  25. func NewDeleteDatastoreHandler(
  26. config *config.Config,
  27. decoderValidator shared.RequestDecoderValidator,
  28. writer shared.ResultWriter,
  29. ) *DeleteDatastoreHandler {
  30. return &DeleteDatastoreHandler{
  31. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  32. KubernetesAgentGetter: authz.NewOutOfClusterAgentGetter(config),
  33. }
  34. }
  35. func (h *DeleteDatastoreHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  36. ctx, span := telemetry.NewSpan(r.Context(), "serve-delete-datastore")
  37. defer span.End()
  38. project, _ := ctx.Value(types.ProjectScope).(*models.Project)
  39. datastoreName, reqErr := requestutils.GetURLParamString(r, types.URLParamDatastoreName)
  40. if reqErr != nil {
  41. err := telemetry.Error(ctx, span, nil, "error parsing datastore name")
  42. h.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  43. return
  44. }
  45. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "datastore-name", Value: datastoreName})
  46. datastoreRecord, err := h.Repo().Datastore().GetByProjectIDAndName(ctx, project.ID, datastoreName)
  47. if err != nil {
  48. err = telemetry.Error(ctx, span, err, "datastore record not found")
  49. h.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  50. return
  51. }
  52. if datastoreRecord == nil || datastoreRecord.ID == uuid.Nil {
  53. err = telemetry.Error(ctx, span, nil, "datastore record does not exist")
  54. h.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusNotFound))
  55. return
  56. }
  57. _, err = h.Repo().Datastore().UpdateStatus(ctx, datastoreRecord, models.DatastoreStatus_AwaitingDeletion)
  58. if err != nil {
  59. err = telemetry.Error(ctx, span, err, "error updating datastore status")
  60. h.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  61. return
  62. }
  63. updateReq := connect.NewRequest(&porterv1.UpdateDatastoreRequest{
  64. ProjectId: int64(project.ID),
  65. DatastoreId: datastoreRecord.ID.String(),
  66. })
  67. _, err = h.Config().ClusterControlPlaneClient.UpdateDatastore(ctx, updateReq)
  68. if err != nil {
  69. err := telemetry.Error(ctx, span, err, "error calling ccp update datastore")
  70. h.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  71. return
  72. }
  73. // if the release was deleted by helm without error, mark it as accepted
  74. w.WriteHeader(http.StatusAccepted)
  75. }
  76. // UninstallDatastoreInput is the input type for UninstallDatastore
  77. type UninstallDatastoreInput struct {
  78. ProjectID uint
  79. Name string
  80. CloudProvider string
  81. CloudProviderCredentialIdentifier string
  82. Request *http.Request
  83. }
  84. // UninstallDatastore uninstalls a datastore from a cluster
  85. func (h *DeleteDatastoreHandler) UninstallDatastore(ctx context.Context, inp UninstallDatastoreInput) error {
  86. ctx, span := telemetry.NewSpan(ctx, "uninstall-datastore")
  87. defer span.End()
  88. telemetry.WithAttributes(span,
  89. telemetry.AttributeKV{Key: "project-id", Value: inp.ProjectID},
  90. telemetry.AttributeKV{Key: "name", Value: inp.Name},
  91. telemetry.AttributeKV{Key: "cloud-provider", Value: inp.CloudProvider},
  92. telemetry.AttributeKV{Key: "cloud-provider-credential-identifier", Value: inp.CloudProviderCredentialIdentifier},
  93. )
  94. var datastoreCluster *models.Cluster
  95. clusters, err := h.Repo().Cluster().ListClustersByProjectID(inp.ProjectID)
  96. if err != nil {
  97. return telemetry.Error(ctx, span, err, "unable to get project clusters")
  98. }
  99. for _, cluster := range clusters {
  100. if cluster.CloudProvider == inp.CloudProvider && cluster.CloudProviderCredentialIdentifier == inp.CloudProviderCredentialIdentifier {
  101. datastoreCluster = cluster
  102. }
  103. }
  104. if datastoreCluster == nil {
  105. return telemetry.Error(ctx, span, nil, "unable to find datastore cluster")
  106. }
  107. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "cluster-id", Value: datastoreCluster.ID})
  108. helmAgent, err := h.GetHelmAgent(ctx, inp.Request, datastoreCluster, release.Namespace_ACKSystem)
  109. if err != nil {
  110. return telemetry.Error(ctx, span, err, "unable to get helm client for cluster")
  111. }
  112. _, err = helmAgent.UninstallChart(ctx, inp.Name)
  113. if err != nil {
  114. return telemetry.Error(ctx, span, err, "unable to uninstall chart")
  115. }
  116. return nil
  117. }