plan.go 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  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.MetronomeEnabled || !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.MetronomeEnabled},
  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.MetronomeEnabled || !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.MetronomeEnabled},
  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. // GetUsageDashboardHandler returns an embeddable dashboard to display information related to customer usage.
  87. type GetUsageDashboardHandler struct {
  88. handlers.PorterHandlerReadWriter
  89. }
  90. // NewGetUsageDashboardHandler returns a new GetUsageDashboardHandler
  91. func NewGetUsageDashboardHandler(
  92. config *config.Config,
  93. decoderValidator shared.RequestDecoderValidator,
  94. writer shared.ResultWriter,
  95. ) *GetUsageDashboardHandler {
  96. return &GetUsageDashboardHandler{
  97. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  98. }
  99. }
  100. func (c *GetUsageDashboardHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  101. ctx, span := telemetry.NewSpan(r.Context(), "serve-usage-dashboard")
  102. defer span.End()
  103. proj, _ := ctx.Value(types.ProjectScope).(*models.Project)
  104. if !c.Config().BillingManager.MetronomeEnabled || !proj.GetFeatureFlag(models.MetronomeEnabled, c.Config().LaunchDarklyClient) {
  105. c.WriteResult(w, r, "")
  106. telemetry.WithAttributes(span,
  107. telemetry.AttributeKV{Key: "metronome-config-exists", Value: c.Config().BillingManager.MetronomeEnabled},
  108. telemetry.AttributeKV{Key: "metronome-enabled", Value: proj.GetFeatureFlag(models.MetronomeEnabled, c.Config().LaunchDarklyClient)},
  109. )
  110. return
  111. }
  112. telemetry.WithAttributes(span,
  113. telemetry.AttributeKV{Key: "metronome-enabled", Value: true},
  114. telemetry.AttributeKV{Key: "usage-id", Value: proj.UsageID},
  115. )
  116. request := &types.EmbeddableDashboardRequest{}
  117. if ok := c.DecodeAndValidate(w, r, request); !ok {
  118. err := telemetry.Error(ctx, span, nil, "error decoding embeddable usage dashboard request")
  119. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  120. return
  121. }
  122. credits, err := c.Config().BillingManager.MetronomeClient.GetCustomerDashboard(ctx, proj.UsageID, request.DashboardType, request.Options, request.ColorOverrides)
  123. if err != nil {
  124. err := telemetry.Error(ctx, span, err, "error getting customer dashboard")
  125. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  126. return
  127. }
  128. c.WriteResult(w, r, credits)
  129. }
  130. // ListCustomerUsageHandler returns customer usage aggregations like CPU and RAM hours.
  131. type ListCustomerUsageHandler struct {
  132. handlers.PorterHandlerReadWriter
  133. }
  134. // NewListCustomerUsageHandler returns a new ListCustomerUsageHandler
  135. func NewListCustomerUsageHandler(
  136. config *config.Config,
  137. decoderValidator shared.RequestDecoderValidator,
  138. writer shared.ResultWriter,
  139. ) *ListCustomerUsageHandler {
  140. return &ListCustomerUsageHandler{
  141. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  142. }
  143. }
  144. func (c *ListCustomerUsageHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  145. ctx, span := telemetry.NewSpan(r.Context(), "serve-list-customer-usage")
  146. defer span.End()
  147. proj, _ := ctx.Value(types.ProjectScope).(*models.Project)
  148. telemetry.WithAttributes(span,
  149. telemetry.AttributeKV{Key: "metronome-config-exists", Value: c.Config().BillingManager.MetronomeEnabled},
  150. telemetry.AttributeKV{Key: "metronome-enabled", Value: proj.GetFeatureFlag(models.MetronomeEnabled, c.Config().LaunchDarklyClient)},
  151. telemetry.AttributeKV{Key: "usage-id", Value: proj.UsageID},
  152. )
  153. if !c.Config().BillingManager.MetronomeEnabled || !proj.GetFeatureFlag(models.MetronomeEnabled, c.Config().LaunchDarklyClient) {
  154. c.WriteResult(w, r, "")
  155. return
  156. }
  157. req := &types.ListCustomerUsageRequest{}
  158. if ok := c.DecodeAndValidate(w, r, req); !ok {
  159. err := telemetry.Error(ctx, span, nil, "error decoding list customer usage request")
  160. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  161. return
  162. }
  163. usage, err := c.Config().BillingManager.MetronomeClient.ListCustomerUsage(ctx, proj.UsageID, req.StartingOn, req.EndingBefore, req.WindowSize, req.CurrentPeriod)
  164. if err != nil {
  165. err := telemetry.Error(ctx, span, err, "error listing customer usage")
  166. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  167. return
  168. }
  169. c.WriteResult(w, r, usage)
  170. }