handlers.go 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  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. w.Write(WrapData(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. w.Write(WrapData(carbonEstimates, nil))
  52. }
  53. func (a *Accesses) computeAssetsFromCostmodel(window opencost.Window, filterString string) (*opencost.AssetSet, error) {
  54. assetSet, err := a.Model.ComputeAssets(*window.Start(), *window.End())
  55. if err != nil {
  56. return nil, fmt.Errorf("error computing asset set: %s", err)
  57. }
  58. var filter opencost.AssetMatcher
  59. if filterString == "" {
  60. filter = &matcher.AllPass[opencost.Asset]{}
  61. } else {
  62. parser := assetfilter.NewAssetFilterParser()
  63. tree, errParse := parser.Parse(filterString)
  64. if errParse != nil {
  65. return nil, fmt.Errorf("err parsing filter '%s': %v", ast.ToPreOrderShortString(tree), errParse)
  66. }
  67. compiler := opencost.NewAssetMatchCompiler()
  68. var err error
  69. filter, err = compiler.Compile(tree)
  70. if err != nil {
  71. return nil, fmt.Errorf("err compiling filter '%s': %v", ast.ToPreOrderShortString(tree), err)
  72. }
  73. }
  74. if filter == nil {
  75. return nil, fmt.Errorf("unexpected nil filter")
  76. }
  77. for key, asset := range assetSet.Assets {
  78. if !filter.Matches(asset) {
  79. delete(assetSet.Assets, key)
  80. }
  81. }
  82. return assetSet, nil
  83. }