credential.go 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  1. package datastore
  2. import (
  3. "net/http"
  4. "connectrpc.com/connect"
  5. "github.com/google/uuid"
  6. porterv1 "github.com/porter-dev/api-contracts/generated/go/porter/v1"
  7. "github.com/porter-dev/porter/api/server/authz"
  8. "github.com/porter-dev/porter/api/server/handlers"
  9. "github.com/porter-dev/porter/api/server/shared"
  10. "github.com/porter-dev/porter/api/server/shared/apierrors"
  11. "github.com/porter-dev/porter/api/server/shared/config"
  12. "github.com/porter-dev/porter/api/server/shared/requestutils"
  13. "github.com/porter-dev/porter/api/types"
  14. "github.com/porter-dev/porter/internal/datastore"
  15. "github.com/porter-dev/porter/internal/models"
  16. "github.com/porter-dev/porter/internal/telemetry"
  17. )
  18. // GetDatastoreCredentialResponse describes the datastore credential response body
  19. type GetDatastoreCredentialResponse struct {
  20. // Credential is the credential that has been retrieved for this datastore
  21. Credential datastore.Credential `json:"credential"`
  22. }
  23. // GetDatastoreCredentialHandler is a struct for retrieving credentials for datastore
  24. type GetDatastoreCredentialHandler struct {
  25. handlers.PorterHandlerReadWriter
  26. authz.KubernetesAgentGetter
  27. }
  28. // NewGetDatastoreCredentialHandler returns a GetDatastoreCredentialHandler
  29. func NewGetDatastoreCredentialHandler(
  30. config *config.Config,
  31. decoderValidator shared.RequestDecoderValidator,
  32. writer shared.ResultWriter,
  33. ) *GetDatastoreCredentialHandler {
  34. return &GetDatastoreCredentialHandler{
  35. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  36. KubernetesAgentGetter: authz.NewOutOfClusterAgentGetter(config),
  37. }
  38. }
  39. // ServeHTTP retrieves the credentials for a datastore
  40. func (c *GetDatastoreCredentialHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  41. ctx, span := telemetry.NewSpan(r.Context(), "serve-get-datastore-credential")
  42. defer span.End()
  43. project, _ := ctx.Value(types.ProjectScope).(*models.Project)
  44. if project.ID == 0 {
  45. err := telemetry.Error(ctx, span, nil, "project not found")
  46. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  47. return
  48. }
  49. projectId := int64(project.ID)
  50. var resp GetDatastoreCredentialResponse
  51. datastoreName, reqErr := requestutils.GetURLParamString(r, types.URLParamDatastoreName)
  52. if reqErr != nil {
  53. err := telemetry.Error(ctx, span, nil, "error parsing datastore name")
  54. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  55. return
  56. }
  57. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "datastore-name", Value: datastoreName})
  58. datastoreRecord, err := c.Repo().Datastore().GetByProjectIDAndName(ctx, project.ID, datastoreName)
  59. if err != nil {
  60. err = telemetry.Error(ctx, span, err, "datastore record not found")
  61. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  62. return
  63. }
  64. if datastoreRecord == nil || datastoreRecord.ID == uuid.Nil {
  65. err = telemetry.Error(ctx, span, nil, "datastore record does not exist")
  66. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  67. return
  68. }
  69. message := porterv1.DatastoreCredentialRequest{
  70. ProjectId: projectId,
  71. DatastoreId: datastoreRecord.ID.String(),
  72. }
  73. req := connect.NewRequest(&message)
  74. ccpResp, err := c.Config().ClusterControlPlaneClient.DatastoreCredential(ctx, req)
  75. if err != nil {
  76. err = telemetry.Error(ctx, span, err, "error getting datastore credential")
  77. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  78. return
  79. }
  80. if ccpResp == nil || ccpResp.Msg == nil {
  81. err = telemetry.Error(ctx, span, nil, "datastore credential not found")
  82. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  83. return
  84. }
  85. resp = GetDatastoreCredentialResponse{
  86. Credential: datastore.Credential{
  87. Host: ccpResp.Msg.Credential.Host,
  88. Port: int(ccpResp.Msg.Credential.Port),
  89. Username: ccpResp.Msg.Credential.Username,
  90. Password: ccpResp.Msg.Credential.Password,
  91. DatabaseName: ccpResp.Msg.Credential.DatabaseName,
  92. },
  93. }
  94. c.WriteResult(w, r, resp)
  95. }