cloud_test.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462
  1. package test
  2. import (
  3. "fmt"
  4. "math"
  5. "os"
  6. "strings"
  7. "testing"
  8. "time"
  9. "github.com/kubecost/cost-model/pkg/cloud"
  10. "github.com/kubecost/cost-model/pkg/clustercache"
  11. "github.com/kubecost/cost-model/pkg/config"
  12. "github.com/kubecost/cost-model/pkg/costmodel"
  13. "github.com/kubecost/cost-model/pkg/costmodel/clusters"
  14. appsv1 "k8s.io/api/apps/v1"
  15. v1 "k8s.io/api/core/v1"
  16. )
  17. const (
  18. providerIDMap = "spec.providerID"
  19. nameMap = "metadata.name"
  20. labelMapFoo = "metadata.labels.foo"
  21. )
  22. func TestRegionValueFromMapField(t *testing.T) {
  23. wantRegion := "useast"
  24. wantpid := strings.ToLower("/subscriptions/0bd50fdf-c923-4e1e-850c-196dd3dcc5d3/resourceGroups/MC_test_test_eastus/providers/Microsoft.Compute/virtualMachines/aks-agentpool-20139558-0")
  25. providerIDWant := wantRegion + "," + wantpid
  26. n := &v1.Node{}
  27. n.Spec.ProviderID = "azure:///subscriptions/0bd50fdf-c923-4e1e-850c-196dd3dcc5d3/resourceGroups/MC_test_test_eastus/providers/Microsoft.Compute/virtualMachines/aks-agentpool-20139558-0"
  28. n.Labels = make(map[string]string)
  29. n.Labels[v1.LabelZoneRegion] = wantRegion
  30. got := cloud.NodeValueFromMapField(providerIDMap, n, true)
  31. if got != providerIDWant {
  32. t.Errorf("Assert on '%s' want '%s' got '%s'", providerIDMap, providerIDWant, got)
  33. }
  34. }
  35. func TestTransformedValueFromMapField(t *testing.T) {
  36. providerIDWant := "i-05445591e0d182d42"
  37. n := &v1.Node{}
  38. n.Spec.ProviderID = "aws:///us-east-1a/i-05445591e0d182d42"
  39. got := cloud.NodeValueFromMapField(providerIDMap, n, false)
  40. if got != providerIDWant {
  41. t.Errorf("Assert on '%s' want '%s' got '%s'", providerIDMap, providerIDWant, got)
  42. }
  43. providerIDWant2 := strings.ToLower("/subscriptions/0bd50fdf-c923-4e1e-850c-196dd3dcc5d3/resourceGroups/MC_test_test_eastus/providers/Microsoft.Compute/virtualMachines/aks-agentpool-20139558-0")
  44. n2 := &v1.Node{}
  45. n2.Spec.ProviderID = "azure:///subscriptions/0bd50fdf-c923-4e1e-850c-196dd3dcc5d3/resourceGroups/MC_test_test_eastus/providers/Microsoft.Compute/virtualMachines/aks-agentpool-20139558-0"
  46. got2 := cloud.NodeValueFromMapField(providerIDMap, n2, false)
  47. if got2 != providerIDWant2 {
  48. t.Errorf("Assert on '%s' want '%s' got '%s'", providerIDMap, providerIDWant2, got2)
  49. }
  50. providerIDWant3 := strings.ToLower("/subscriptions/0bd50fdf-c923-4e1e-850c-196dd3dcc5d3/resourceGroups/mc_testspot_testspot_eastus/providers/Microsoft.Compute/virtualMachineScaleSets/aks-nodepool1-19213364-vmss/virtualMachines/0")
  51. n3 := &v1.Node{}
  52. n3.Spec.ProviderID = "azure:///subscriptions/0bd50fdf-c923-4e1e-850c-196dd3dcc5d3/resourceGroups/mc_testspot_testspot_eastus/providers/Microsoft.Compute/virtualMachineScaleSets/aks-nodepool1-19213364-vmss/virtualMachines/0"
  53. got3 := cloud.NodeValueFromMapField(providerIDMap, n3, false)
  54. if got3 != providerIDWant3 {
  55. t.Errorf("Assert on '%s' want '%s' got '%s'", providerIDMap, providerIDWant3, got3)
  56. }
  57. }
  58. func TestNodeValueFromMapField(t *testing.T) {
  59. providerIDWant := "providerid"
  60. nameWant := "gke-standard-cluster-1-pool-1-91dc432d-cg69"
  61. labelFooWant := "labelfoo"
  62. n := &v1.Node{}
  63. n.Spec.ProviderID = providerIDWant
  64. n.Name = nameWant
  65. n.Labels = make(map[string]string)
  66. n.Labels["foo"] = labelFooWant
  67. got := cloud.NodeValueFromMapField(providerIDMap, n, false)
  68. if got != providerIDWant {
  69. t.Errorf("Assert on '%s' want '%s' got '%s'", providerIDMap, providerIDWant, got)
  70. }
  71. got = cloud.NodeValueFromMapField(nameMap, n, false)
  72. if got != nameWant {
  73. t.Errorf("Assert on '%s' want '%s' got '%s'", nameMap, nameWant, got)
  74. }
  75. got = cloud.NodeValueFromMapField(labelMapFoo, n, false)
  76. if got != labelFooWant {
  77. t.Errorf("Assert on '%s' want '%s' got '%s'", labelMapFoo, labelFooWant, got)
  78. }
  79. }
  80. func TestNodePriceFromCSV(t *testing.T) {
  81. providerIDWant := "providerid"
  82. nameWant := "gke-standard-cluster-1-pool-1-91dc432d-cg69"
  83. labelFooWant := "labelfoo"
  84. confMan := config.NewConfigFileManager(&config.ConfigFileManagerOpts{
  85. LocalConfigPath: "./",
  86. })
  87. n := &v1.Node{}
  88. n.Spec.ProviderID = providerIDWant
  89. n.Name = nameWant
  90. n.Labels = make(map[string]string)
  91. n.Labels["foo"] = labelFooWant
  92. wantPrice := "0.1337"
  93. c := &cloud.CSVProvider{
  94. CSVLocation: "../configs/pricing_schema.csv",
  95. CustomProvider: &cloud.CustomProvider{
  96. Config: cloud.NewProviderConfig(confMan, "../configs/default.json"),
  97. },
  98. }
  99. c.DownloadPricingData()
  100. k := c.GetKey(n.Labels, n)
  101. resN, err := c.NodePricing(k)
  102. if err != nil {
  103. t.Errorf("Error in NodePricing: %s", err.Error())
  104. } else {
  105. gotPrice := resN.Cost
  106. if gotPrice != wantPrice {
  107. t.Errorf("Wanted price '%s' got price '%s'", wantPrice, gotPrice)
  108. }
  109. }
  110. unknownN := &v1.Node{}
  111. unknownN.Spec.ProviderID = providerIDWant
  112. unknownN.Name = "unknownname"
  113. unknownN.Labels = make(map[string]string)
  114. unknownN.Labels["foo"] = labelFooWant
  115. unknownN.Labels["topology.kubernetes.io/region"] = "fakeregion"
  116. k2 := c.GetKey(unknownN.Labels, unknownN)
  117. resN2, _ := c.NodePricing(k2)
  118. if resN2 != nil {
  119. t.Errorf("CSV provider should return nil on missing node")
  120. }
  121. c2 := &cloud.CSVProvider{
  122. CSVLocation: "../configs/fake.csv",
  123. CustomProvider: &cloud.CustomProvider{
  124. Config: cloud.NewProviderConfig(confMan, "../configs/default.json"),
  125. },
  126. }
  127. k3 := c.GetKey(n.Labels, n)
  128. resN3, _ := c2.NodePricing(k3)
  129. if resN3 != nil {
  130. t.Errorf("CSV provider should return nil on missing csv")
  131. }
  132. }
  133. func TestNodePriceFromCSVWithRegion(t *testing.T) {
  134. providerIDWant := "gke-standard-cluster-1-pool-1-91dc432d-cg69"
  135. nameWant := "foo"
  136. labelFooWant := "labelfoo"
  137. confMan := config.NewConfigFileManager(&config.ConfigFileManagerOpts{
  138. LocalConfigPath: "./",
  139. })
  140. n := &v1.Node{}
  141. n.Spec.ProviderID = providerIDWant
  142. n.Name = nameWant
  143. n.Labels = make(map[string]string)
  144. n.Labels["foo"] = labelFooWant
  145. n.Labels[v1.LabelZoneRegion] = "regionone"
  146. wantPrice := "0.1337"
  147. n2 := &v1.Node{}
  148. n2.Spec.ProviderID = providerIDWant
  149. n2.Name = nameWant
  150. n2.Labels = make(map[string]string)
  151. n2.Labels["foo"] = labelFooWant
  152. n2.Labels[v1.LabelZoneRegion] = "regiontwo"
  153. wantPrice2 := "0.1338"
  154. n3 := &v1.Node{}
  155. n3.Spec.ProviderID = providerIDWant
  156. n3.Name = nameWant
  157. n3.Labels = make(map[string]string)
  158. n3.Labels["foo"] = labelFooWant
  159. n3.Labels[v1.LabelZoneRegion] = "fakeregion"
  160. wantPrice3 := "0.1339"
  161. c := &cloud.CSVProvider{
  162. CSVLocation: "../configs/pricing_schema_region.csv",
  163. CustomProvider: &cloud.CustomProvider{
  164. Config: cloud.NewProviderConfig(confMan, "../configs/default.json"),
  165. },
  166. }
  167. c.DownloadPricingData()
  168. k := c.GetKey(n.Labels, n)
  169. resN, err := c.NodePricing(k)
  170. if err != nil {
  171. t.Errorf("Error in NodePricing: %s", err.Error())
  172. } else {
  173. gotPrice := resN.Cost
  174. if gotPrice != wantPrice {
  175. t.Errorf("Wanted price '%s' got price '%s'", wantPrice, gotPrice)
  176. }
  177. }
  178. k2 := c.GetKey(n2.Labels, n2)
  179. resN2, err := c.NodePricing(k2)
  180. if err != nil {
  181. t.Errorf("Error in NodePricing: %s", err.Error())
  182. } else {
  183. gotPrice := resN2.Cost
  184. if gotPrice != wantPrice2 {
  185. t.Errorf("Wanted price '%s' got price '%s'", wantPrice2, gotPrice)
  186. }
  187. }
  188. k3 := c.GetKey(n3.Labels, n3)
  189. resN3, err := c.NodePricing(k3)
  190. if err != nil {
  191. t.Errorf("Error in NodePricing: %s", err.Error())
  192. } else {
  193. gotPrice := resN3.Cost
  194. if gotPrice != wantPrice3 {
  195. t.Errorf("Wanted price '%s' got price '%s'", wantPrice3, gotPrice)
  196. }
  197. }
  198. unknownN := &v1.Node{}
  199. unknownN.Spec.ProviderID = "fake providerID"
  200. unknownN.Name = "unknownname"
  201. unknownN.Labels = make(map[string]string)
  202. unknownN.Labels["topology.kubernetes.io/region"] = "fakeregion"
  203. unknownN.Labels["foo"] = labelFooWant
  204. k4 := c.GetKey(unknownN.Labels, unknownN)
  205. resN4, _ := c.NodePricing(k4)
  206. if resN4 != nil {
  207. t.Errorf("CSV provider should return nil on missing node, instead returned %+v", resN4)
  208. }
  209. c2 := &cloud.CSVProvider{
  210. CSVLocation: "../configs/fake.csv",
  211. CustomProvider: &cloud.CustomProvider{
  212. Config: cloud.NewProviderConfig(confMan, "../configs/default.json"),
  213. },
  214. }
  215. k5 := c.GetKey(n.Labels, n)
  216. resN5, _ := c2.NodePricing(k5)
  217. if resN5 != nil {
  218. t.Errorf("CSV provider should return nil on missing csv")
  219. }
  220. }
  221. type FakeCache struct {
  222. nodes []*v1.Node
  223. clustercache.ClusterCache
  224. }
  225. func (f FakeCache) GetAllNodes() []*v1.Node {
  226. return f.nodes
  227. }
  228. func (f FakeCache) GetAllDaemonSets() []*appsv1.DaemonSet {
  229. return nil
  230. }
  231. func NewFakeNodeCache(nodes []*v1.Node) FakeCache {
  232. return FakeCache{
  233. nodes: nodes,
  234. }
  235. }
  236. type FakeClusterMap struct {
  237. clusters.ClusterMap
  238. }
  239. func TestNodePriceFromCSVWithBadConfig(t *testing.T) {
  240. os.Setenv("CONFIG_PATH", "../config")
  241. confMan := config.NewConfigFileManager(&config.ConfigFileManagerOpts{
  242. LocalConfigPath: "./",
  243. })
  244. c := &cloud.CSVProvider{
  245. CSVLocation: "../configs/pricing_schema_case.csv",
  246. CustomProvider: &cloud.CustomProvider{
  247. Config: cloud.NewProviderConfig(confMan, "invalid.json"),
  248. },
  249. }
  250. c.DownloadPricingData()
  251. n := &v1.Node{}
  252. n.Spec.ProviderID = "fake"
  253. n.Name = "nameWant"
  254. n.Labels = make(map[string]string)
  255. n.Labels["foo"] = "labelFooWant"
  256. n.Labels[v1.LabelZoneRegion] = "regionone"
  257. fc := NewFakeNodeCache([]*v1.Node{n})
  258. fm := FakeClusterMap{}
  259. d, _ := time.ParseDuration("1m")
  260. model := costmodel.NewCostModel(nil, nil, fc, fm, d)
  261. _, err := model.GetNodeCost(c)
  262. if err != nil {
  263. t.Errorf("Error in node pricing: %s", err)
  264. }
  265. }
  266. func TestSourceMatchesFromCSV(t *testing.T) {
  267. os.Setenv("CONFIG_PATH", "../configs")
  268. confMan := config.NewConfigFileManager(&config.ConfigFileManagerOpts{
  269. LocalConfigPath: "./",
  270. })
  271. c := &cloud.CSVProvider{
  272. CSVLocation: "../configs/pricing_schema_case.csv",
  273. CustomProvider: &cloud.CustomProvider{
  274. Config: cloud.NewProviderConfig(confMan, "/default.json"),
  275. },
  276. }
  277. c.DownloadPricingData()
  278. n := &v1.Node{}
  279. n.Spec.ProviderID = "fake"
  280. n.Name = "nameWant"
  281. n.Labels = make(map[string]string)
  282. n.Labels["foo"] = "labelFooWant"
  283. n.Labels[v1.LabelZoneRegion] = "regionone"
  284. n2 := &v1.Node{}
  285. n2.Spec.ProviderID = "azure:///subscriptions/123a7sd-asd-1234-578a9-123abcdef/resourceGroups/case_12_STaGe_TeSt7/providers/Microsoft.Compute/virtualMachineScaleSets/vmss-agent-worker0-12stagetest7-ezggnore/virtualMachines/7"
  286. n2.Labels = make(map[string]string)
  287. n2.Labels[v1.LabelZoneRegion] = "eastus2"
  288. n2.Labels["foo"] = "labelFooWant"
  289. k := c.GetKey(n2.Labels, n2)
  290. resN, err := c.NodePricing(k)
  291. if err != nil {
  292. t.Errorf("Error in NodePricing: %s", err.Error())
  293. } else {
  294. wantPrice := "0.13370357"
  295. gotPrice := resN.Cost
  296. if gotPrice != wantPrice {
  297. t.Errorf("Wanted price '%s' got price '%s'", wantPrice, gotPrice)
  298. }
  299. }
  300. n3 := &v1.Node{}
  301. n3.Spec.ProviderID = "fake"
  302. n3.Name = "nameWant"
  303. n3.Labels = make(map[string]string)
  304. n.Labels[v1.LabelZoneRegion] = "eastus2"
  305. n.Labels[v1.LabelInstanceType] = "Standard_F32s_v2"
  306. fc := NewFakeNodeCache([]*v1.Node{n, n2, n3})
  307. fm := FakeClusterMap{}
  308. d, _ := time.ParseDuration("1m")
  309. model := costmodel.NewCostModel(nil, nil, fc, fm, d)
  310. _, err = model.GetNodeCost(c)
  311. if err != nil {
  312. t.Errorf("Error in node pricing: %s", err)
  313. }
  314. p, err := model.GetPricingSourceCounts()
  315. if err != nil {
  316. t.Errorf("Error in pricing source counts: %s", err)
  317. } else if p.TotalNodes != 3 {
  318. t.Errorf("Wanted 3 nodes got %d", p.TotalNodes)
  319. }
  320. if p.PricingTypeCounts[""] != 1 {
  321. t.Errorf("Wanted 1 default match got %d: %+v", p.PricingTypeCounts[""], p.PricingTypeCounts)
  322. }
  323. if p.PricingTypeCounts["csvExact"] != 1 {
  324. t.Errorf("Wanted 1 exact match got %d: %+v", p.PricingTypeCounts["csvExact"], p.PricingTypeCounts)
  325. }
  326. if p.PricingTypeCounts["csvClass"] != 1 {
  327. t.Errorf("Wanted 1 class match got %d: %+v", p.PricingTypeCounts["csvClass"], p.PricingTypeCounts)
  328. }
  329. }
  330. func TestNodePriceFromCSVWithCase(t *testing.T) {
  331. n := &v1.Node{}
  332. n.Spec.ProviderID = "azure:///subscriptions/123a7sd-asd-1234-578a9-123abcdef/resourceGroups/case_12_STaGe_TeSt7/providers/Microsoft.Compute/virtualMachineScaleSets/vmss-agent-worker0-12stagetest7-ezggnore/virtualMachines/7"
  333. n.Labels = make(map[string]string)
  334. n.Labels[v1.LabelZoneRegion] = "eastus2"
  335. wantPrice := "0.13370357"
  336. confMan := config.NewConfigFileManager(&config.ConfigFileManagerOpts{
  337. LocalConfigPath: "./",
  338. })
  339. c := &cloud.CSVProvider{
  340. CSVLocation: "../configs/pricing_schema_case.csv",
  341. CustomProvider: &cloud.CustomProvider{
  342. Config: cloud.NewProviderConfig(confMan, "../configs/default.json"),
  343. },
  344. }
  345. c.DownloadPricingData()
  346. k := c.GetKey(n.Labels, n)
  347. resN, err := c.NodePricing(k)
  348. if err != nil {
  349. t.Errorf("Error in NodePricing: %s", err.Error())
  350. } else {
  351. gotPrice := resN.Cost
  352. if gotPrice != wantPrice {
  353. t.Errorf("Wanted price '%s' got price '%s'", wantPrice, gotPrice)
  354. }
  355. }
  356. }
  357. func TestNodePriceFromCSVByClass(t *testing.T) {
  358. n := &v1.Node{}
  359. n.Spec.ProviderID = "fakeproviderid"
  360. n.Labels = make(map[string]string)
  361. n.Labels[v1.LabelZoneRegion] = "eastus2"
  362. n.Labels[v1.LabelInstanceType] = "Standard_F32s_v2"
  363. wantpricefloat := 0.13370357
  364. wantPrice := fmt.Sprintf("%f", (math.Round(wantpricefloat*1000000) / 1000000))
  365. confMan := config.NewConfigFileManager(&config.ConfigFileManagerOpts{
  366. LocalConfigPath: "./",
  367. })
  368. c := &cloud.CSVProvider{
  369. CSVLocation: "../configs/pricing_schema_case.csv",
  370. CustomProvider: &cloud.CustomProvider{
  371. Config: cloud.NewProviderConfig(confMan, "../configs/default.json"),
  372. },
  373. }
  374. c.DownloadPricingData()
  375. k := c.GetKey(n.Labels, n)
  376. resN, err := c.NodePricing(k)
  377. if err != nil {
  378. t.Errorf("Error in NodePricing: %s", err.Error())
  379. } else {
  380. gotPrice := resN.Cost
  381. if gotPrice != wantPrice {
  382. t.Errorf("Wanted price '%s' got price '%s'", wantPrice, gotPrice)
  383. }
  384. }
  385. n2 := &v1.Node{}
  386. n2.Spec.ProviderID = "fakeproviderid"
  387. n2.Labels = make(map[string]string)
  388. n2.Labels[v1.LabelZoneRegion] = "fakeregion"
  389. n2.Labels[v1.LabelInstanceType] = "Standard_F32s_v2"
  390. k2 := c.GetKey(n2.Labels, n)
  391. c.DownloadPricingData()
  392. resN2, err := c.NodePricing(k2)
  393. if resN2 != nil {
  394. t.Errorf("CSV provider should return nil on missing node, instead returned %+v", resN2)
  395. }
  396. }