job_status.go 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  1. package porter_app
  2. import (
  3. "net/http"
  4. "github.com/porter-dev/porter/api/server/authz"
  5. "github.com/porter-dev/porter/api/server/handlers"
  6. "github.com/porter-dev/porter/api/server/shared"
  7. "github.com/porter-dev/porter/api/server/shared/apierrors"
  8. "github.com/porter-dev/porter/api/server/shared/config"
  9. "github.com/porter-dev/porter/api/server/shared/requestutils"
  10. "github.com/porter-dev/porter/api/types"
  11. "github.com/porter-dev/porter/internal/deployment_target"
  12. "github.com/porter-dev/porter/internal/kubernetes"
  13. "github.com/porter-dev/porter/internal/models"
  14. "github.com/porter-dev/porter/internal/telemetry"
  15. )
  16. // JobStatusHandler is the handler for GET /apps/jobs
  17. type JobStatusHandler struct {
  18. handlers.PorterHandlerReadWriter
  19. authz.KubernetesAgentGetter
  20. }
  21. // NewJobStatusHandler returns a new JobStatusHandler
  22. func NewJobStatusHandler(
  23. config *config.Config,
  24. decoderValidator shared.RequestDecoderValidator,
  25. writer shared.ResultWriter,
  26. ) *JobStatusHandler {
  27. return &JobStatusHandler{
  28. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  29. KubernetesAgentGetter: authz.NewOutOfClusterAgentGetter(config),
  30. }
  31. }
  32. // JobStatusRequest is the expected format for a request body on GET /apps/jobs
  33. type JobStatusRequest struct {
  34. DeploymentTargetID string `schema:"deployment_target_id"`
  35. JobName string `schema:"job_name"`
  36. }
  37. func (c *JobStatusHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  38. ctx, span := telemetry.NewSpan(r.Context(), "serve-job-status")
  39. defer span.End()
  40. request := &JobStatusRequest{}
  41. if ok := c.DecodeAndValidate(w, r, request); !ok {
  42. err := telemetry.Error(ctx, span, nil, "invalid request")
  43. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  44. return
  45. }
  46. cluster, _ := ctx.Value(types.ClusterScope).(*models.Cluster)
  47. project, _ := ctx.Value(types.ProjectScope).(*models.Project)
  48. name, reqErr := requestutils.GetURLParamString(r, types.URLParamPorterAppName)
  49. if reqErr != nil {
  50. err := telemetry.Error(ctx, span, reqErr, "invalid porter app name")
  51. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  52. return
  53. }
  54. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "app-name", Value: name})
  55. if request.DeploymentTargetID == "" {
  56. err := telemetry.Error(ctx, span, nil, "must provide deployment target id")
  57. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  58. return
  59. }
  60. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "deployment-target-id", Value: request.DeploymentTargetID})
  61. deploymentTarget, err := deployment_target.DeploymentTargetDetails(ctx, deployment_target.DeploymentTargetDetailsInput{
  62. ProjectID: int64(project.ID),
  63. ClusterID: int64(cluster.ID),
  64. DeploymentTargetID: request.DeploymentTargetID,
  65. CCPClient: c.Config().ClusterControlPlaneClient,
  66. })
  67. if err != nil {
  68. err := telemetry.Error(ctx, span, err, "error getting deployment target details")
  69. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  70. return
  71. }
  72. namespace := deploymentTarget.Namespace
  73. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "namespace", Value: namespace})
  74. agent, err := c.GetAgent(r, cluster, "")
  75. if err != nil {
  76. err = telemetry.Error(ctx, span, err, "unable to get agent")
  77. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  78. return
  79. }
  80. labels := []kubernetes.Label{
  81. {
  82. Key: "porter.run/deployment-target-id",
  83. Val: request.DeploymentTargetID,
  84. },
  85. {
  86. Key: "porter.run/app-name",
  87. Val: name,
  88. },
  89. }
  90. if request.JobName != "" {
  91. labels = append(labels, kubernetes.Label{
  92. Key: "porter.run/service-name",
  93. Val: request.JobName,
  94. })
  95. }
  96. jobs, err := agent.ListJobsByLabel(namespace, labels...)
  97. if err != nil {
  98. err = telemetry.Error(ctx, span, err, "error listing jobs")
  99. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  100. return
  101. }
  102. c.WriteResult(w, r, jobs)
  103. }