cluster_status.go 2.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283
  1. package cluster
  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/types"
  13. "github.com/porter-dev/porter/internal/models"
  14. "github.com/porter-dev/porter/internal/telemetry"
  15. )
  16. type ClusterStatusHandler struct {
  17. handlers.PorterHandlerReadWriter
  18. authz.KubernetesAgentGetter
  19. }
  20. func NewClusterStatusHandler(
  21. config *config.Config,
  22. decoderValidator shared.RequestDecoderValidator,
  23. writer shared.ResultWriter,
  24. ) *ClusterStatusHandler {
  25. return &ClusterStatusHandler{
  26. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  27. KubernetesAgentGetter: authz.NewOutOfClusterAgentGetter(config),
  28. }
  29. }
  30. type ClusterStatusResponse struct {
  31. ProjectID int `json:"project_id"`
  32. ClusterID int `json:"cluster_id"`
  33. Phase string `json:"phase"`
  34. IsInfrastructureReady bool `json:"is_infrastructure_ready"`
  35. IsControlPlaneReady bool `json:"is_control_plane_ready"`
  36. }
  37. func (c *ClusterStatusHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  38. ctx, span := telemetry.NewSpan(r.Context(), "serve-cluster-status")
  39. defer span.End()
  40. cluster, _ := ctx.Value(types.ClusterScope).(*models.Cluster)
  41. req := connect.NewRequest(&porterv1.ClusterStatusRequest{
  42. ProjectId: int64(cluster.ProjectID),
  43. ClusterId: int64(cluster.ID),
  44. })
  45. status, err := c.Config().ClusterControlPlaneClient.ClusterStatus(ctx, req)
  46. if err != nil {
  47. err := fmt.Errorf("unable to retrieve status for cluster: %w", err)
  48. err = telemetry.Error(ctx, span, err, err.Error())
  49. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  50. return
  51. }
  52. if status.Msg == nil {
  53. err := fmt.Errorf("unable to parse status for cluster: %w", err)
  54. err = telemetry.Error(ctx, span, err, err.Error())
  55. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  56. return
  57. }
  58. statusResp := status.Msg
  59. resp := ClusterStatusResponse{
  60. ProjectID: int(statusResp.ProjectId),
  61. ClusterID: int(statusResp.ClusterId),
  62. Phase: statusResp.Phase,
  63. IsInfrastructureReady: statusResp.InfrastructureStatus,
  64. IsControlPlaneReady: statusResp.ControlPlaneStatus,
  65. }
  66. telemetry.WithAttributes(span,
  67. telemetry.AttributeKV{Key: "cluster-phase", Value: statusResp.Phase},
  68. telemetry.AttributeKV{Key: "cluster-infra-status", Value: statusResp.InfrastructureStatus},
  69. telemetry.AttributeKV{Key: "cluster-control-plane-status", Value: statusResp.ControlPlaneStatus},
  70. )
  71. c.WriteResult(w, r, resp)
  72. w.WriteHeader(http.StatusOK)
  73. }