pod_status.go 4.2 KB

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