plan.go 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. package billing
  2. import (
  3. "net/http"
  4. "github.com/porter-dev/porter/api/server/handlers"
  5. "github.com/porter-dev/porter/api/server/shared"
  6. "github.com/porter-dev/porter/api/server/shared/apierrors"
  7. "github.com/porter-dev/porter/api/server/shared/config"
  8. "github.com/porter-dev/porter/api/types"
  9. "github.com/porter-dev/porter/internal/models"
  10. "github.com/porter-dev/porter/internal/telemetry"
  11. )
  12. // ListPlansHandler is a handler for getting customer plans
  13. type ListPlansHandler struct {
  14. handlers.PorterHandlerWriter
  15. }
  16. // NewListPlansHandler will create a new ListPlansHandler
  17. func NewListPlansHandler(
  18. config *config.Config,
  19. writer shared.ResultWriter,
  20. ) *ListPlansHandler {
  21. return &ListPlansHandler{
  22. PorterHandlerWriter: handlers.NewDefaultPorterHandler(config, nil, writer),
  23. }
  24. }
  25. func (c *ListPlansHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  26. ctx, span := telemetry.NewSpan(r.Context(), "serve-list-plans")
  27. defer span.End()
  28. proj, _ := ctx.Value(types.ProjectScope).(*models.Project)
  29. if !c.Config().BillingManager.MetronomeConfigLoaded || !proj.GetFeatureFlag(models.MetronomeEnabled, c.Config().LaunchDarklyClient) {
  30. c.WriteResult(w, r, "")
  31. telemetry.WithAttributes(span,
  32. telemetry.AttributeKV{Key: "metronome-config-exists", Value: c.Config().BillingManager.MetronomeConfigLoaded},
  33. telemetry.AttributeKV{Key: "metronome-enabled", Value: proj.GetFeatureFlag(models.MetronomeEnabled, c.Config().LaunchDarklyClient)},
  34. )
  35. return
  36. }
  37. telemetry.WithAttributes(span,
  38. telemetry.AttributeKV{Key: "metronome-enabled", Value: true},
  39. telemetry.AttributeKV{Key: "usage-id", Value: proj.UsageID},
  40. )
  41. plan, err := c.Config().BillingManager.MetronomeClient.ListCustomerPlan(ctx, proj.UsageID)
  42. if err != nil {
  43. err := telemetry.Error(ctx, span, err, "error listing plans")
  44. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  45. return
  46. }
  47. c.WriteResult(w, r, plan)
  48. }
  49. // ListCreditsHandler is a handler for getting available credits
  50. type ListCreditsHandler struct {
  51. handlers.PorterHandlerWriter
  52. }
  53. // NewListCreditsHandler will create a new ListCreditsHandler
  54. func NewListCreditsHandler(
  55. config *config.Config,
  56. writer shared.ResultWriter,
  57. ) *ListCreditsHandler {
  58. return &ListCreditsHandler{
  59. PorterHandlerWriter: handlers.NewDefaultPorterHandler(config, nil, writer),
  60. }
  61. }
  62. func (c *ListCreditsHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  63. ctx, span := telemetry.NewSpan(r.Context(), "serve-list-credits")
  64. defer span.End()
  65. proj, _ := ctx.Value(types.ProjectScope).(*models.Project)
  66. if !c.Config().BillingManager.MetronomeConfigLoaded || !proj.GetFeatureFlag(models.MetronomeEnabled, c.Config().LaunchDarklyClient) {
  67. c.WriteResult(w, r, "")
  68. telemetry.WithAttributes(span,
  69. telemetry.AttributeKV{Key: "metronome-config-exists", Value: c.Config().BillingManager.MetronomeConfigLoaded},
  70. telemetry.AttributeKV{Key: "metronome-enabled", Value: proj.GetFeatureFlag(models.MetronomeEnabled, c.Config().LaunchDarklyClient)},
  71. )
  72. return
  73. }
  74. credits, err := c.Config().BillingManager.MetronomeClient.ListCustomerCredits(ctx, proj.UsageID)
  75. if err != nil {
  76. err := telemetry.Error(ctx, span, err, "error listing credits")
  77. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  78. return
  79. }
  80. telemetry.WithAttributes(span,
  81. telemetry.AttributeKV{Key: "metronome-enabled", Value: true},
  82. telemetry.AttributeKV{Key: "usage-id", Value: proj.UsageID},
  83. )
  84. c.WriteResult(w, r, credits)
  85. }
  86. // ListCustomerUsageHandler returns customer usage aggregations like CPU and RAM hours.
  87. type ListCustomerUsageHandler struct {
  88. handlers.PorterHandlerReadWriter
  89. }
  90. // NewListCustomerUsageHandler returns a new ListCustomerUsageHandler
  91. func NewListCustomerUsageHandler(
  92. config *config.Config,
  93. decoderValidator shared.RequestDecoderValidator,
  94. writer shared.ResultWriter,
  95. ) *ListCustomerUsageHandler {
  96. return &ListCustomerUsageHandler{
  97. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  98. }
  99. }
  100. func (c *ListCustomerUsageHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  101. ctx, span := telemetry.NewSpan(r.Context(), "serve-list-customer-usage")
  102. defer span.End()
  103. proj, _ := ctx.Value(types.ProjectScope).(*models.Project)
  104. telemetry.WithAttributes(span,
  105. telemetry.AttributeKV{Key: "metronome-config-exists", Value: c.Config().BillingManager.MetronomeConfigLoaded},
  106. telemetry.AttributeKV{Key: "metronome-enabled", Value: proj.GetFeatureFlag(models.MetronomeEnabled, c.Config().LaunchDarklyClient)},
  107. telemetry.AttributeKV{Key: "usage-id", Value: proj.UsageID},
  108. )
  109. if !c.Config().BillingManager.MetronomeConfigLoaded || !proj.GetFeatureFlag(models.MetronomeEnabled, c.Config().LaunchDarklyClient) {
  110. c.WriteResult(w, r, "")
  111. return
  112. }
  113. req := &types.ListCustomerUsageRequest{}
  114. if ok := c.DecodeAndValidate(w, r, req); !ok {
  115. err := telemetry.Error(ctx, span, nil, "error decoding list customer usage request")
  116. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  117. return
  118. }
  119. usage, err := c.Config().BillingManager.MetronomeClient.ListCustomerUsage(ctx, proj.UsageID, req.StartingOn, req.EndingBefore, req.WindowSize, req.CurrentPeriod)
  120. if err != nil {
  121. err := telemetry.Error(ctx, span, err, "error listing customer usage")
  122. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  123. return
  124. }
  125. c.WriteResult(w, r, usage)
  126. }
  127. // ListCustomerCostsHandler returns customer usage aggregations like CPU and RAM hours.
  128. type ListCustomerCostsHandler struct {
  129. handlers.PorterHandlerReadWriter
  130. }
  131. // NewListCustomerCostsHandler returns a new ListCustomerCostsHandler
  132. func NewListCustomerCostsHandler(
  133. config *config.Config,
  134. decoderValidator shared.RequestDecoderValidator,
  135. writer shared.ResultWriter,
  136. ) *ListCustomerCostsHandler {
  137. return &ListCustomerCostsHandler{
  138. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  139. }
  140. }
  141. func (c *ListCustomerCostsHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  142. ctx, span := telemetry.NewSpan(r.Context(), "serve-list-customer-costs")
  143. defer span.End()
  144. proj, _ := ctx.Value(types.ProjectScope).(*models.Project)
  145. telemetry.WithAttributes(span,
  146. telemetry.AttributeKV{Key: "metronome-config-exists", Value: c.Config().BillingManager.MetronomeConfigLoaded},
  147. telemetry.AttributeKV{Key: "metronome-enabled", Value: proj.GetFeatureFlag(models.MetronomeEnabled, c.Config().LaunchDarklyClient)},
  148. telemetry.AttributeKV{Key: "usage-id", Value: proj.UsageID},
  149. )
  150. if !c.Config().BillingManager.MetronomeConfigLoaded || !proj.GetFeatureFlag(models.MetronomeEnabled, c.Config().LaunchDarklyClient) {
  151. c.WriteResult(w, r, "")
  152. return
  153. }
  154. req := &types.ListCustomerCostsRequest{}
  155. if ok := c.DecodeAndValidate(w, r, req); !ok {
  156. err := telemetry.Error(ctx, span, nil, "error decoding list customer costs request")
  157. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  158. return
  159. }
  160. usage, err := c.Config().BillingManager.MetronomeClient.ListCustomerCosts(ctx, proj.UsageID, req.StartingOn, req.EndingBefore, req.Limit)
  161. if err != nil {
  162. err := telemetry.Error(ctx, span, err, "error listing customer costs")
  163. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  164. return
  165. }
  166. c.WriteResult(w, r, usage)
  167. }