pod_status.go 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. package porter_app
  2. import (
  3. "fmt"
  4. "net/http"
  5. "connectrpc.com/connect"
  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/models"
  15. "github.com/porter-dev/porter/internal/telemetry"
  16. v1 "k8s.io/api/core/v1"
  17. )
  18. // PodStatusHandler is the handler for GET /apps/pods
  19. type PodStatusHandler struct {
  20. handlers.PorterHandlerReadWriter
  21. authz.KubernetesAgentGetter
  22. }
  23. // NewPodStatusHandler returns a new PodStatusHandler
  24. func NewPodStatusHandler(
  25. config *config.Config,
  26. decoderValidator shared.RequestDecoderValidator,
  27. writer shared.ResultWriter,
  28. ) *PodStatusHandler {
  29. return &PodStatusHandler{
  30. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  31. KubernetesAgentGetter: authz.NewOutOfClusterAgentGetter(config),
  32. }
  33. }
  34. // PodStatusRequest is the expected format for a request body on GET /apps/pods
  35. type PodStatusRequest struct {
  36. DeploymentTargetID string `schema:"deployment_target_id"`
  37. ServiceName string `schema:"service"`
  38. }
  39. func (c *PodStatusHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  40. ctx, span := telemetry.NewSpan(r.Context(), "serve-pod-status")
  41. defer span.End()
  42. request := &PodStatusRequest{}
  43. if ok := c.DecodeAndValidate(w, r, request); !ok {
  44. err := telemetry.Error(ctx, span, nil, "invalid request")
  45. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  46. return
  47. }
  48. appName, reqErr := requestutils.GetURLParamString(r, types.URLParamPorterAppName)
  49. if reqErr != nil {
  50. err := telemetry.Error(ctx, span, reqErr, "porter app name not found in request")
  51. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  52. return
  53. }
  54. cluster, _ := r.Context().Value(types.ClusterScope).(*models.Cluster)
  55. project, _ := r.Context().Value(types.ProjectScope).(*models.Project)
  56. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "service-name", Value: request.ServiceName}, telemetry.AttributeKV{Key: "app-name", Value: appName})
  57. if request.DeploymentTargetID == "" {
  58. err := telemetry.Error(ctx, span, nil, "must provide deployment target id")
  59. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  60. return
  61. }
  62. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "deployment-target-id", Value: request.DeploymentTargetID})
  63. deploymentTargetDetailsReq := connect.NewRequest(&porterv1.DeploymentTargetDetailsRequest{
  64. ProjectId: int64(project.ID),
  65. DeploymentTargetId: request.DeploymentTargetID,
  66. })
  67. deploymentTargetDetailsResp, err := c.Config().ClusterControlPlaneClient.DeploymentTargetDetails(ctx, deploymentTargetDetailsReq)
  68. if err != nil {
  69. err := telemetry.Error(ctx, span, err, "error getting deployment target details from cluster control plane client")
  70. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  71. return
  72. }
  73. if deploymentTargetDetailsResp == nil || deploymentTargetDetailsResp.Msg == nil {
  74. err := telemetry.Error(ctx, span, err, "deployment target details resp is nil")
  75. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  76. return
  77. }
  78. if deploymentTargetDetailsResp.Msg.ClusterId != int64(cluster.ID) {
  79. err := telemetry.Error(ctx, span, err, "deployment target details resp cluster id does not match cluster id")
  80. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  81. return
  82. }
  83. namespace := deploymentTargetDetailsResp.Msg.Namespace
  84. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "namespace", Value: namespace})
  85. agent, err := c.GetAgent(r, cluster, "")
  86. if err != nil {
  87. err = telemetry.Error(ctx, span, err, "unable to get agent")
  88. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  89. return
  90. }
  91. pods := []v1.Pod{}
  92. var selectors string
  93. if request.ServiceName == "" {
  94. selectors = fmt.Sprintf("porter.run/deployment-target-id=%s,porter.run/app-name=%s", request.DeploymentTargetID, appName)
  95. } else {
  96. selectors = fmt.Sprintf("porter.run/service-name=%s,porter.run/deployment-target-id=%s,porter.run/app-name=%s", request.ServiceName, request.DeploymentTargetID, appName)
  97. }
  98. podsList, err := agent.GetPodsByLabel(selectors, namespace)
  99. if err != nil {
  100. err = telemetry.Error(ctx, span, err, "unable to get pods by label")
  101. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  102. return
  103. }
  104. pods = append(pods, podsList.Items...)
  105. c.WriteResult(w, r, pods)
  106. }