compliance_checks.go 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  1. package cluster
  2. import (
  3. "net/http"
  4. "connectrpc.com/connect"
  5. porterv1 "github.com/porter-dev/api-contracts/generated/go/porter/v1"
  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/types"
  11. "github.com/porter-dev/porter/internal/compliance"
  12. "github.com/porter-dev/porter/internal/models"
  13. "github.com/porter-dev/porter/internal/telemetry"
  14. )
  15. // ListComplianceChecksHandler is the handler for /compliance/checks
  16. type ListComplianceChecksHandler struct {
  17. handlers.PorterHandlerReadWriter
  18. }
  19. // NewListComplianceChecksHandler returns a new ListComplianceChecksHandler
  20. func NewListComplianceChecksHandler(
  21. config *config.Config,
  22. decoderValidator shared.RequestDecoderValidator,
  23. writer shared.ResultWriter,
  24. ) *ListComplianceChecksHandler {
  25. return &ListComplianceChecksHandler{
  26. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  27. }
  28. }
  29. // ListComplianceChecksRequest is the expected format for a request to /compliance/checks
  30. type ListComplianceChecksRequest struct {
  31. Vendor compliance.Vendor `schema:"vendor"`
  32. }
  33. // ListComplianceChecksResponse is the expected format for a response from /compliance/checks
  34. type ListComplianceChecksResponse struct {
  35. CheckGroups []compliance.CheckGroup `json:"check_groups,omitempty"`
  36. VendorChecks []compliance.VendorComplianceCheck `json:"vendor_checks,omitempty"`
  37. }
  38. // ServeHTTP retrieves the evaluated compliance checks for a cluster
  39. func (c *ListComplianceChecksHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  40. ctx, span := telemetry.NewSpan(r.Context(), "serve-compliance-checks")
  41. defer span.End()
  42. project, _ := ctx.Value(types.ProjectScope).(*models.Project)
  43. cluster, _ := ctx.Value(types.ClusterScope).(*models.Cluster)
  44. request := &ListComplianceChecksRequest{}
  45. if ok := c.DecodeAndValidate(w, r, request); !ok {
  46. err := telemetry.Error(ctx, span, nil, "error decoding request")
  47. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  48. return
  49. }
  50. var vendor porterv1.EnumComplianceVendor
  51. if request.Vendor != "" {
  52. switch request.Vendor {
  53. case compliance.Vendor_Vanta:
  54. vendor = porterv1.EnumComplianceVendor_ENUM_COMPLIANCE_VENDOR_VANTA
  55. default:
  56. err := telemetry.Error(ctx, span, nil, "invalid vendor")
  57. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  58. return
  59. }
  60. }
  61. req := connect.NewRequest(&porterv1.ContractComplianceChecksRequest{
  62. ProjectId: int64(project.ID),
  63. ClusterId: int64(cluster.ID),
  64. Vendor: vendor,
  65. })
  66. ccpResp, err := c.Config().ClusterControlPlaneClient.ContractComplianceChecks(ctx, req)
  67. if err != nil {
  68. err := telemetry.Error(ctx, span, err, "error calling ccp for contract compliance checks")
  69. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  70. return
  71. }
  72. if ccpResp == nil {
  73. err := telemetry.Error(ctx, span, err, "ccp resp is nil")
  74. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  75. return
  76. }
  77. if ccpResp.Msg == nil {
  78. err := telemetry.Error(ctx, span, err, "ccp resp msg is nil")
  79. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  80. return
  81. }
  82. cgs, err := compliance.CheckGroupsFromProto(ctx, ccpResp.Msg.CheckGroups)
  83. if err != nil {
  84. err := telemetry.Error(ctx, span, err, "error converting compliance check groups from proto")
  85. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  86. return
  87. }
  88. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "num-check-groups", Value: len(cgs)})
  89. vendorChecks, err := compliance.VendorCheckGroupsFromProto(ctx, ccpResp.Msg.VendorChecks)
  90. if err != nil {
  91. err := telemetry.Error(ctx, span, err, "error converting vendor compliance check groups from proto")
  92. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  93. return
  94. }
  95. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "num-vendor-checks", Value: len(vendorChecks)})
  96. resp := &ListComplianceChecksResponse{
  97. CheckGroups: cgs,
  98. VendorChecks: vendorChecks,
  99. }
  100. c.WriteResult(w, r, resp)
  101. w.WriteHeader(http.StatusOK)
  102. }