2
0

handlers.go 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107
  1. package costmodel
  2. import (
  3. "fmt"
  4. "net/http"
  5. "github.com/julienschmidt/httprouter"
  6. assetfilter "github.com/opencost/opencost/core/pkg/filter/asset"
  7. "github.com/opencost/opencost/core/pkg/filter/ast"
  8. "github.com/opencost/opencost/core/pkg/filter/matcher"
  9. "github.com/opencost/opencost/core/pkg/opencost"
  10. "github.com/opencost/opencost/core/pkg/util/httputil"
  11. "github.com/opencost/opencost/pkg/carbon"
  12. "github.com/opencost/opencost/pkg/env"
  13. )
  14. // ComputeAllocationHandler returns the assets from the CostModel.
  15. func (a *Accesses) ComputeAssetsHandler(w http.ResponseWriter, r *http.Request, ps httprouter.Params) {
  16. w.Header().Set("Content-Type", "application/json")
  17. qp := httputil.NewQueryParams(r.URL.Query())
  18. // Window is a required field describing the window of time over which to
  19. // compute allocation data.
  20. window, err := opencost.ParseWindowWithOffset(qp.Get("window", ""), env.GetParsedUTCOffset())
  21. if err != nil {
  22. http.Error(w, fmt.Sprintf("Invalid 'window' parameter: %s", err), http.StatusBadRequest)
  23. return
  24. }
  25. filterString := qp.Get("filter", "")
  26. assetSet, err := a.ComputeAssetsFromCostmodel(window, filterString)
  27. if err != nil {
  28. http.Error(w, fmt.Sprintf("Error getting assets: %s", err), http.StatusInternalServerError)
  29. return
  30. }
  31. WriteData(w, assetSet, nil)
  32. }
  33. // ComputeAllocationHandler returns the assets from the CostModel.
  34. func (a *Accesses) ComputeAssetsCarbonHandler(w http.ResponseWriter, r *http.Request, ps httprouter.Params) {
  35. w.Header().Set("Content-Type", "application/json")
  36. qp := httputil.NewQueryParams(r.URL.Query())
  37. // Window is a required field describing the window of time over which to
  38. // compute allocation data.
  39. window, err := opencost.ParseWindowWithOffset(qp.Get("window", ""), env.GetParsedUTCOffset())
  40. if err != nil {
  41. http.Error(w, fmt.Sprintf("Invalid 'window' parameter: %s", err), http.StatusBadRequest)
  42. return
  43. }
  44. filterString := qp.Get("filter", "")
  45. assetSet, err := a.ComputeAssetsFromCostmodel(window, filterString)
  46. if err != nil {
  47. http.Error(w, fmt.Sprintf("Error getting assets: %s", err), http.StatusInternalServerError)
  48. return
  49. }
  50. carbonEstimates, err := carbon.RelateCarbonAssets(assetSet)
  51. if err != nil {
  52. http.Error(w, fmt.Sprintf("Error relating carbon assets: %s", err), http.StatusInternalServerError)
  53. return
  54. }
  55. WriteData(w, carbonEstimates, nil)
  56. }
  57. func (a *Accesses) ComputeAssetsFromCostmodel(window opencost.Window, filterString string) (*opencost.AssetSet, error) {
  58. assetSet, err := a.Model.ComputeAssets(*window.Start(), *window.End())
  59. if err != nil {
  60. return nil, fmt.Errorf("error computing asset set: %s", err)
  61. }
  62. var filter opencost.AssetMatcher
  63. if filterString == "" {
  64. filter = &matcher.AllPass[opencost.Asset]{}
  65. } else {
  66. parser := assetfilter.NewAssetFilterParser()
  67. tree, errParse := parser.Parse(filterString)
  68. if errParse != nil {
  69. return nil, fmt.Errorf("err parsing filter '%s': %v", ast.ToPreOrderShortString(tree), errParse)
  70. }
  71. compiler := opencost.NewAssetMatchCompiler()
  72. var err error
  73. filter, err = compiler.Compile(tree)
  74. if err != nil {
  75. return nil, fmt.Errorf("err compiling filter '%s': %v", ast.ToPreOrderShortString(tree), err)
  76. }
  77. }
  78. if filter == nil {
  79. return nil, fmt.Errorf("unexpected nil filter")
  80. }
  81. for key, asset := range assetSet.Assets {
  82. if !filter.Matches(asset) {
  83. delete(assetSet.Assets, key)
  84. }
  85. }
  86. return assetSet, nil
  87. }