pod_status.go 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137
  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. deploymentTargetName := request.DeploymentTargetName
  63. if request.DeploymentTargetName == "" && request.DeploymentTargetID == "" {
  64. defaultDeploymentTarget, err := defaultDeploymentTarget(ctx, defaultDeploymentTargetInput{
  65. ProjectID: project.ID,
  66. ClusterID: cluster.ID,
  67. ClusterControlPlaneClient: c.Config().ClusterControlPlaneClient,
  68. })
  69. if err != nil {
  70. err := telemetry.Error(ctx, span, err, "error getting default deployment target")
  71. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  72. return
  73. }
  74. deploymentTargetName = defaultDeploymentTarget.Name
  75. }
  76. telemetry.WithAttributes(span,
  77. telemetry.AttributeKV{Key: "deployment-target-id", Value: request.DeploymentTargetID},
  78. telemetry.AttributeKV{Key: "deployment-target-name", Value: request.DeploymentTargetName},
  79. )
  80. deploymentTarget, err := deployment_target.DeploymentTargetDetails(ctx, deployment_target.DeploymentTargetDetailsInput{
  81. ProjectID: int64(project.ID),
  82. ClusterID: int64(cluster.ID),
  83. DeploymentTargetID: request.DeploymentTargetID,
  84. DeploymentTargetName: deploymentTargetName,
  85. CCPClient: c.Config().ClusterControlPlaneClient,
  86. })
  87. if err != nil {
  88. err := telemetry.Error(ctx, span, err, "error getting deployment target details")
  89. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  90. return
  91. }
  92. namespace := deploymentTarget.Namespace
  93. telemetry.WithAttributes(span,
  94. telemetry.AttributeKV{Key: "namespace", Value: namespace},
  95. telemetry.AttributeKV{Key: "deployment-target-id", Value: deploymentTarget.ID},
  96. )
  97. agent, err := c.GetAgent(r, cluster, "")
  98. if err != nil {
  99. err = telemetry.Error(ctx, span, err, "unable to get agent")
  100. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  101. return
  102. }
  103. pods := []v1.Pod{}
  104. var selectors string
  105. if request.ServiceName == "" {
  106. selectors = fmt.Sprintf("porter.run/deployment-target-id=%s,porter.run/app-name=%s", deploymentTarget.ID, appName)
  107. } else {
  108. selectors = fmt.Sprintf("porter.run/service-name=%s,porter.run/deployment-target-id=%s,porter.run/app-name=%s", deploymentTarget.ID, request.DeploymentTargetID, appName)
  109. }
  110. podsList, err := agent.GetPodsByLabel(selectors, namespace)
  111. if err != nil {
  112. err = telemetry.Error(ctx, span, err, "unable to get pods by label")
  113. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  114. return
  115. }
  116. pods = append(pods, podsList.Items...)
  117. c.WriteResult(w, r, pods)
  118. }