pod_status.go 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  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. DeploymentTargetName string `schema:"deployment_target_name"`
  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,
  57. telemetry.AttributeKV{Key: "service-name", Value: request.ServiceName},
  58. telemetry.AttributeKV{Key: "app-name", Value: appName},
  59. telemetry.AttributeKV{Key: "input-deployment-target-id", Value: request.DeploymentTargetID},
  60. telemetry.AttributeKV{Key: "input-deployment-target-name", Value: request.DeploymentTargetName},
  61. )
  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. DeploymentTargetName: request.DeploymentTargetName,
  67. CCPClient: c.Config().ClusterControlPlaneClient,
  68. })
  69. if err != nil {
  70. err := telemetry.Error(ctx, span, err, "error getting deployment target details")
  71. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  72. return
  73. }
  74. namespace := deploymentTarget.Namespace
  75. telemetry.WithAttributes(span,
  76. telemetry.AttributeKV{Key: "namespace", Value: namespace},
  77. telemetry.AttributeKV{Key: "deployment-target-id", Value: deploymentTarget.ID},
  78. )
  79. agent, err := c.GetAgent(r, cluster, "")
  80. if err != nil {
  81. err = telemetry.Error(ctx, span, err, "unable to get agent")
  82. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  83. return
  84. }
  85. pods := []v1.Pod{}
  86. var selectors string
  87. if request.ServiceName == "" {
  88. selectors = fmt.Sprintf("porter.run/deployment-target-id=%s,porter.run/app-name=%s", deploymentTarget.ID, appName)
  89. } else {
  90. selectors = fmt.Sprintf("porter.run/service-name=%s,porter.run/deployment-target-id=%s,porter.run/app-name=%s", deploymentTarget.ID, request.DeploymentTargetID, appName)
  91. }
  92. podsList, err := agent.GetPodsByLabel(selectors, namespace)
  93. if err != nil {
  94. err = telemetry.Error(ctx, span, err, "unable to get pods by label")
  95. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  96. return
  97. }
  98. pods = append(pods, podsList.Items...)
  99. c.WriteResult(w, r, pods)
  100. }