costmodel.go 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  1. package costmodel
  2. import (
  3. "context"
  4. "fmt"
  5. "net/http"
  6. "time"
  7. "github.com/julienschmidt/httprouter"
  8. "github.com/opencost/opencost/core/pkg/util/apiutil"
  9. "github.com/opencost/opencost/pkg/cloud/models"
  10. "github.com/opencost/opencost/pkg/cloud/provider"
  11. "github.com/opencost/opencost/pkg/customcost"
  12. "github.com/prometheus/client_golang/prometheus/promhttp"
  13. "github.com/rs/cors"
  14. "github.com/opencost/opencost/core/pkg/errors"
  15. "github.com/opencost/opencost/core/pkg/log"
  16. "github.com/opencost/opencost/core/pkg/version"
  17. "github.com/opencost/opencost/pkg/costmodel"
  18. "github.com/opencost/opencost/pkg/env"
  19. "github.com/opencost/opencost/pkg/filemanager"
  20. "github.com/opencost/opencost/pkg/metrics"
  21. )
  22. func Execute(conf *Config) error {
  23. log.Infof("Starting cost-model version %s", version.FriendlyVersion())
  24. if conf == nil {
  25. conf = DefaultConfig()
  26. }
  27. conf.log()
  28. router := httprouter.New()
  29. var a *costmodel.Accesses
  30. var cp models.Provider
  31. if conf.KubernetesEnabled {
  32. a = costmodel.Initialize(router)
  33. err := StartExportWorker(context.Background(), a.Model)
  34. if err != nil {
  35. log.Errorf("couldn't start CSV export worker: %v", err)
  36. }
  37. // Register OpenCost Specific Endpoints
  38. router.GET("/allocation", a.ComputeAllocationHandler)
  39. router.GET("/allocation/summary", a.ComputeAllocationHandlerSummary)
  40. router.GET("/assets", a.ComputeAssetsHandler)
  41. if conf.CarbonEstimatesEnabled {
  42. router.GET("/assets/carbon", a.ComputeAssetsCarbonHandler)
  43. }
  44. // set cloud provider for cloud cost
  45. cp = a.CloudProvider
  46. }
  47. if conf.CloudCostEnabled {
  48. var providerConfig models.ProviderConfig
  49. if cp != nil {
  50. providerConfig = provider.ExtractConfigFromProviders(cp)
  51. }
  52. costmodel.InitializeCloudCost(router, providerConfig)
  53. }
  54. var customCostPipelineService *customcost.PipelineService
  55. if conf.CloudCostEnabled {
  56. customCostPipelineService = costmodel.InitializeCustomCost(router)
  57. }
  58. // this endpoint is intentionally left out of the "if env.IsCustomCostEnabled()" conditional; in the handler, it is
  59. // valid for CustomCostPipelineService to be nil
  60. router.GET("/customCost/status", customCostPipelineService.GetCustomCostStatusHandler())
  61. apiutil.ApplyContainerDiagnosticEndpoints(router)
  62. rootMux := http.NewServeMux()
  63. rootMux.Handle("/", router)
  64. rootMux.Handle("/metrics", promhttp.Handler())
  65. telemetryHandler := metrics.ResponseMetricMiddleware(rootMux)
  66. handler := cors.AllowAll().Handler(telemetryHandler)
  67. return http.ListenAndServe(fmt.Sprint(":", conf.Port), errors.PanicHandlerMiddleware(handler))
  68. }
  69. func StartExportWorker(ctx context.Context, model costmodel.AllocationModel) error {
  70. exportPath := env.GetExportCSVFile()
  71. if exportPath == "" {
  72. log.Infof("%s is not set, CSV export is disabled", env.ExportCSVFile)
  73. return nil
  74. }
  75. fm, err := filemanager.NewFileManager(exportPath)
  76. if err != nil {
  77. return fmt.Errorf("could not create file manager: %v", err)
  78. }
  79. go func() {
  80. log.Info("Starting CSV exporter worker...")
  81. // perform first update immediately
  82. nextRunAt := time.Now()
  83. for {
  84. select {
  85. case <-ctx.Done():
  86. return
  87. case <-time.After(nextRunAt.Sub(time.Now())):
  88. err := costmodel.UpdateCSV(ctx, fm, model, env.GetExportCSVLabelsAll(), env.GetExportCSVLabelsList())
  89. if err != nil {
  90. // it's background worker, log error and carry on, maybe next time it will work
  91. log.Errorf("Error updating CSV: %s", err)
  92. }
  93. now := time.Now().UTC()
  94. // next launch is at 00:10 UTC tomorrow
  95. // extra 10 minutes is to let prometheus to collect all the data for the previous day
  96. nextRunAt = time.Date(now.Year(), now.Month(), now.Day(), 0, 10, 0, 0, now.Location()).AddDate(0, 0, 1)
  97. }
  98. }
  99. }()
  100. return nil
  101. }