ingest.go 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. // NewGetUsageDashboardHandler returns a new GetUsageDashboardHandler
  2. package billing
  3. import (
  4. "fmt"
  5. "net/http"
  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/models"
  12. "github.com/porter-dev/porter/internal/telemetry"
  13. )
  14. // IngestEventsHandler is a handler for ingesting billing events
  15. type IngestEventsHandler struct {
  16. handlers.PorterHandlerReadWriter
  17. }
  18. // NewIngestEventsHandler returns a new IngestEventsHandler
  19. func NewIngestEventsHandler(
  20. config *config.Config,
  21. decoderValidator shared.RequestDecoderValidator,
  22. writer shared.ResultWriter,
  23. ) *IngestEventsHandler {
  24. return &IngestEventsHandler{
  25. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  26. }
  27. }
  28. func (c *IngestEventsHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  29. ctx, span := telemetry.NewSpan(r.Context(), "serve-ingest-events")
  30. defer span.End()
  31. proj, _ := ctx.Value(types.ProjectScope).(*models.Project)
  32. if !c.Config().BillingManager.MetronomeConfigLoaded || !proj.GetFeatureFlag(models.MetronomeEnabled, c.Config().LaunchDarklyClient) {
  33. c.WriteResult(w, r, "")
  34. telemetry.WithAttributes(span,
  35. telemetry.AttributeKV{Key: "metronome-config-exists", Value: c.Config().BillingManager.MetronomeConfigLoaded},
  36. telemetry.AttributeKV{Key: "metronome-enabled", Value: proj.GetFeatureFlag(models.MetronomeEnabled, c.Config().LaunchDarklyClient)},
  37. telemetry.AttributeKV{Key: "porter-cloud-enabled", Value: proj.EnableSandbox},
  38. )
  39. return
  40. }
  41. telemetry.WithAttributes(span,
  42. telemetry.AttributeKV{Key: "metronome-enabled", Value: true},
  43. telemetry.AttributeKV{Key: "usage-id", Value: proj.UsageID},
  44. )
  45. ingestEventsRequest := struct {
  46. Events []types.BillingEvent `json:"billing_events"`
  47. }{}
  48. if ok := c.DecodeAndValidate(w, r, &ingestEventsRequest); !ok {
  49. err := telemetry.Error(ctx, span, nil, "error decoding ingest events request")
  50. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  51. return
  52. }
  53. telemetry.WithAttributes(span,
  54. telemetry.AttributeKV{Key: "usage-events-count", Value: len(ingestEventsRequest.Events)},
  55. )
  56. // For Porter Cloud events, we apend a prefix to avoid collisions before sending to Metronome
  57. if proj.EnableSandbox {
  58. for i := range ingestEventsRequest.Events {
  59. ingestEventsRequest.Events[i].CustomerID = fmt.Sprintf("porter-cloud-%s", ingestEventsRequest.Events[i].CustomerID)
  60. }
  61. }
  62. err := c.Config().BillingManager.MetronomeClient.IngestEvents(ctx, ingestEventsRequest.Events)
  63. if err != nil {
  64. err := telemetry.Error(ctx, span, err, "error ingesting events")
  65. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  66. return
  67. }
  68. c.WriteResult(w, r, "")
  69. }