cloud_test.go 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629
  1. package test
  2. import (
  3. "fmt"
  4. "math"
  5. "os"
  6. "strings"
  7. "testing"
  8. "time"
  9. "github.com/opencost/opencost/pkg/cloud/provider"
  10. "github.com/opencost/opencost/pkg/clustercache"
  11. "github.com/opencost/opencost/pkg/config"
  12. "github.com/opencost/opencost/pkg/costmodel"
  13. "github.com/opencost/opencost/pkg/costmodel/clusters"
  14. appsv1 "k8s.io/api/apps/v1"
  15. v1 "k8s.io/api/core/v1"
  16. "k8s.io/apimachinery/pkg/api/resource"
  17. )
  18. const (
  19. providerIDMap = "spec.providerID"
  20. nameMap = "metadata.name"
  21. labelMapFoo = "metadata.labels.foo"
  22. labelMapFooBar = "metadata.labels.foo.bar"
  23. )
  24. func TestRegionValueFromMapField(t *testing.T) {
  25. wantRegion := "useast"
  26. wantpid := strings.ToLower("/subscriptions/0bd50fdf-c923-4e1e-850c-196dd3dcc5d3/resourceGroups/MC_test_test_eastus/providers/Microsoft.Compute/virtualMachines/aks-agentpool-20139558-0")
  27. providerIDWant := wantRegion + "," + wantpid
  28. n := &v1.Node{}
  29. n.Spec.ProviderID = "azure:///subscriptions/0bd50fdf-c923-4e1e-850c-196dd3dcc5d3/resourceGroups/MC_test_test_eastus/providers/Microsoft.Compute/virtualMachines/aks-agentpool-20139558-0"
  30. n.Labels = make(map[string]string)
  31. n.Labels[v1.LabelZoneRegion] = wantRegion
  32. got := provider.NodeValueFromMapField(providerIDMap, n, true)
  33. if got != providerIDWant {
  34. t.Errorf("Assert on '%s' want '%s' got '%s'", providerIDMap, providerIDWant, got)
  35. }
  36. }
  37. func TestTransformedValueFromMapField(t *testing.T) {
  38. providerIDWant := "i-05445591e0d182d42"
  39. n := &v1.Node{}
  40. n.Spec.ProviderID = "aws:///us-east-1a/i-05445591e0d182d42"
  41. got := provider.NodeValueFromMapField(providerIDMap, n, false)
  42. if got != providerIDWant {
  43. t.Errorf("Assert on '%s' want '%s' got '%s'", providerIDMap, providerIDWant, got)
  44. }
  45. providerIDWant2 := strings.ToLower("/subscriptions/0bd50fdf-c923-4e1e-850c-196dd3dcc5d3/resourceGroups/MC_test_test_eastus/providers/Microsoft.Compute/virtualMachines/aks-agentpool-20139558-0")
  46. n2 := &v1.Node{}
  47. n2.Spec.ProviderID = "azure:///subscriptions/0bd50fdf-c923-4e1e-850c-196dd3dcc5d3/resourceGroups/MC_test_test_eastus/providers/Microsoft.Compute/virtualMachines/aks-agentpool-20139558-0"
  48. got2 := provider.NodeValueFromMapField(providerIDMap, n2, false)
  49. if got2 != providerIDWant2 {
  50. t.Errorf("Assert on '%s' want '%s' got '%s'", providerIDMap, providerIDWant2, got2)
  51. }
  52. providerIDWant3 := strings.ToLower("/subscriptions/0bd50fdf-c923-4e1e-850c-196dd3dcc5d3/resourceGroups/mc_testspot_testspot_eastus/providers/Microsoft.Compute/virtualMachineScaleSets/aks-nodepool1-19213364-vmss/virtualMachines/0")
  53. n3 := &v1.Node{}
  54. 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"
  55. got3 := provider.NodeValueFromMapField(providerIDMap, n3, false)
  56. if got3 != providerIDWant3 {
  57. t.Errorf("Assert on '%s' want '%s' got '%s'", providerIDMap, providerIDWant3, got3)
  58. }
  59. }
  60. func TestNodeValueFromMapField(t *testing.T) {
  61. providerIDWant := "providerid"
  62. nameWant := "gke-standard-cluster-1-pool-1-91dc432d-cg69"
  63. labelFooWant := "labelfoo"
  64. n := &v1.Node{}
  65. n.Spec.ProviderID = providerIDWant
  66. n.Name = nameWant
  67. n.Labels = make(map[string]string)
  68. n.Labels["foo"] = labelFooWant
  69. got := provider.NodeValueFromMapField(providerIDMap, n, false)
  70. if got != providerIDWant {
  71. t.Errorf("Assert on '%s' want '%s' got '%s'", providerIDMap, providerIDWant, got)
  72. }
  73. got = provider.NodeValueFromMapField(nameMap, n, false)
  74. if got != nameWant {
  75. t.Errorf("Assert on '%s' want '%s' got '%s'", nameMap, nameWant, got)
  76. }
  77. got = provider.NodeValueFromMapField(labelMapFoo, n, false)
  78. if got != labelFooWant {
  79. t.Errorf("Assert on '%s' want '%s' got '%s'", labelMapFoo, labelFooWant, got)
  80. }
  81. }
  82. func TestPVPriceFromCSV(t *testing.T) {
  83. nameWant := "pvc-08e1f205-d7a9-4430-90fc-7b3965a18c4d"
  84. pv := &v1.PersistentVolume{}
  85. pv.Name = nameWant
  86. confMan := config.NewConfigFileManager(&config.ConfigFileManagerOpts{
  87. LocalConfigPath: "./",
  88. })
  89. wantPrice := "0.1337"
  90. c := &provider.CSVProvider{
  91. CSVLocation: "../configs/pricing_schema_pv.csv",
  92. CustomProvider: &provider.CustomProvider{
  93. Config: provider.NewProviderConfig(confMan, "../configs/default.json"),
  94. },
  95. }
  96. c.DownloadPricingData()
  97. k := c.GetPVKey(pv, make(map[string]string), "")
  98. resPV, err := c.PVPricing(k)
  99. if err != nil {
  100. t.Errorf("Error in NodePricing: %s", err.Error())
  101. } else {
  102. gotPrice := resPV.Cost
  103. if gotPrice != wantPrice {
  104. t.Errorf("Wanted price '%s' got price '%s'", wantPrice, gotPrice)
  105. }
  106. }
  107. }
  108. func TestPVPriceFromCSVStorageClass(t *testing.T) {
  109. nameWant := "pvc-08e1f205-d7a9-4430-90fc-7b3965a18c4d"
  110. storageClassWant := "storageclass0"
  111. pv := &v1.PersistentVolume{}
  112. pv.Name = nameWant
  113. pv.Spec.StorageClassName = storageClassWant
  114. confMan := config.NewConfigFileManager(&config.ConfigFileManagerOpts{
  115. LocalConfigPath: "./",
  116. })
  117. wantPrice := "0.1338"
  118. c := &provider.CSVProvider{
  119. CSVLocation: "../configs/pricing_schema_pv_storageclass.csv",
  120. CustomProvider: &provider.CustomProvider{
  121. Config: provider.NewProviderConfig(confMan, "../configs/default.json"),
  122. },
  123. }
  124. c.DownloadPricingData()
  125. k := c.GetPVKey(pv, make(map[string]string), "")
  126. resPV, err := c.PVPricing(k)
  127. if err != nil {
  128. t.Errorf("Error in NodePricing: %s", err.Error())
  129. } else {
  130. gotPrice := resPV.Cost
  131. if gotPrice != wantPrice {
  132. t.Errorf("Wanted price '%s' got price '%s'", wantPrice, gotPrice)
  133. }
  134. }
  135. }
  136. func TestNodePriceFromCSVWithGPU(t *testing.T) {
  137. providerIDWant := "providerid"
  138. nameWant := "gke-standard-cluster-1-pool-1-91dc432d-cg69"
  139. labelFooWant := "labelfoo"
  140. wantGPU := "2"
  141. confMan := config.NewConfigFileManager(&config.ConfigFileManagerOpts{
  142. LocalConfigPath: "./",
  143. })
  144. n := &v1.Node{}
  145. n.Spec.ProviderID = providerIDWant
  146. n.Name = nameWant
  147. n.Labels = make(map[string]string)
  148. n.Labels["foo"] = labelFooWant
  149. n.Labels["nvidia.com/gpu_type"] = "Quadro_RTX_4000"
  150. n.Status.Capacity = v1.ResourceList{"nvidia.com/gpu": *resource.NewScaledQuantity(2, 0)}
  151. wantPrice := "1.633700"
  152. n2 := &v1.Node{}
  153. n2.Spec.ProviderID = providerIDWant
  154. n2.Name = nameWant
  155. n2.Labels = make(map[string]string)
  156. n2.Labels["foo"] = labelFooWant
  157. n2.Labels["gpu.nvidia.com/class"] = "Quadro_RTX_4001"
  158. n2.Status.Capacity = v1.ResourceList{"nvidia.com/gpu": *resource.NewScaledQuantity(2, 0)}
  159. wantPrice2 := "1.733700"
  160. c := &provider.CSVProvider{
  161. CSVLocation: "../configs/pricing_schema.csv",
  162. CustomProvider: &provider.CustomProvider{
  163. Config: provider.NewProviderConfig(confMan, "../configs/default.json"),
  164. },
  165. }
  166. c.DownloadPricingData()
  167. k := c.GetKey(n.Labels, n)
  168. resN, _, err := c.NodePricing(k)
  169. if err != nil {
  170. t.Errorf("Error in NodePricing: %s", err.Error())
  171. } else {
  172. gotGPU := resN.GPU
  173. gotPrice := resN.Cost
  174. if gotGPU != wantGPU {
  175. t.Errorf("Wanted gpu count '%s' got gpu count '%s'", wantGPU, gotGPU)
  176. }
  177. if gotPrice != wantPrice {
  178. t.Errorf("Wanted price '%s' got price '%s'", wantPrice, gotPrice)
  179. }
  180. }
  181. k2 := c.GetKey(n2.Labels, n2)
  182. resN2, _, err := c.NodePricing(k2)
  183. if err != nil {
  184. t.Errorf("Error in NodePricing: %s", err.Error())
  185. } else {
  186. gotGPU := resN2.GPU
  187. gotPrice := resN2.Cost
  188. if gotGPU != wantGPU {
  189. t.Errorf("Wanted gpu count '%s' got gpu count '%s'", wantGPU, gotGPU)
  190. }
  191. if gotPrice != wantPrice2 {
  192. t.Errorf("Wanted price '%s' got price '%s'", wantPrice2, gotPrice)
  193. }
  194. }
  195. }
  196. func TestNodePriceFromCSVSpecialChar(t *testing.T) {
  197. nameWant := "gke-standard-cluster-1-pool-1-91dc432d-cg69"
  198. confMan := config.NewConfigFileManager(&config.ConfigFileManagerOpts{
  199. LocalConfigPath: "./",
  200. })
  201. n := &v1.Node{}
  202. n.Name = nameWant
  203. n.Labels = make(map[string]string)
  204. n.Labels["<http://metadata.label.servers.com/label|metadata.label.servers.com/label>"] = nameWant
  205. wantPrice := "0.133700"
  206. c := &provider.CSVProvider{
  207. CSVLocation: "../configs/pricing_schema_special_char.csv",
  208. CustomProvider: &provider.CustomProvider{
  209. Config: provider.NewProviderConfig(confMan, "../configs/default.json"),
  210. },
  211. }
  212. c.DownloadPricingData()
  213. k := c.GetKey(n.Labels, n)
  214. resN, _, err := c.NodePricing(k)
  215. if err != nil {
  216. t.Errorf("Error in NodePricing: %s", err.Error())
  217. } else {
  218. gotPrice := resN.Cost
  219. if gotPrice != wantPrice {
  220. t.Errorf("Wanted price '%s' got price '%s'", wantPrice, gotPrice)
  221. }
  222. }
  223. }
  224. func TestNodePriceFromCSV(t *testing.T) {
  225. providerIDWant := "providerid"
  226. nameWant := "gke-standard-cluster-1-pool-1-91dc432d-cg69"
  227. labelFooWant := "labelfoo"
  228. confMan := config.NewConfigFileManager(&config.ConfigFileManagerOpts{
  229. LocalConfigPath: "./",
  230. })
  231. n := &v1.Node{}
  232. n.Spec.ProviderID = providerIDWant
  233. n.Name = nameWant
  234. n.Labels = make(map[string]string)
  235. n.Labels["foo"] = labelFooWant
  236. wantPrice := "0.133700"
  237. c := &provider.CSVProvider{
  238. CSVLocation: "../configs/pricing_schema.csv",
  239. CustomProvider: &provider.CustomProvider{
  240. Config: provider.NewProviderConfig(confMan, "../configs/default.json"),
  241. },
  242. }
  243. c.DownloadPricingData()
  244. k := c.GetKey(n.Labels, n)
  245. resN, _, err := c.NodePricing(k)
  246. if err != nil {
  247. t.Errorf("Error in NodePricing: %s", err.Error())
  248. } else {
  249. gotPrice := resN.Cost
  250. if gotPrice != wantPrice {
  251. t.Errorf("Wanted price '%s' got price '%s'", wantPrice, gotPrice)
  252. }
  253. }
  254. unknownN := &v1.Node{}
  255. unknownN.Spec.ProviderID = providerIDWant
  256. unknownN.Name = "unknownname"
  257. unknownN.Labels = make(map[string]string)
  258. unknownN.Labels["foo"] = labelFooWant
  259. unknownN.Labels["topology.kubernetes.io/region"] = "fakeregion"
  260. k2 := c.GetKey(unknownN.Labels, unknownN)
  261. resN2, _, _ := c.NodePricing(k2)
  262. if resN2 != nil {
  263. t.Errorf("CSV provider should return nil on missing node")
  264. }
  265. c2 := &provider.CSVProvider{
  266. CSVLocation: "../configs/fake.csv",
  267. CustomProvider: &provider.CustomProvider{
  268. Config: provider.NewProviderConfig(confMan, "../configs/default.json"),
  269. },
  270. }
  271. k3 := c.GetKey(n.Labels, n)
  272. resN3, _, _ := c2.NodePricing(k3)
  273. if resN3 != nil {
  274. t.Errorf("CSV provider should return nil on missing csv")
  275. }
  276. }
  277. func TestNodePriceFromCSVWithRegion(t *testing.T) {
  278. providerIDWant := "gke-standard-cluster-1-pool-1-91dc432d-cg69"
  279. nameWant := "foo"
  280. labelFooWant := "labelfoo"
  281. confMan := config.NewConfigFileManager(&config.ConfigFileManagerOpts{
  282. LocalConfigPath: "./",
  283. })
  284. n := &v1.Node{}
  285. n.Spec.ProviderID = providerIDWant
  286. n.Name = nameWant
  287. n.Labels = make(map[string]string)
  288. n.Labels["foo"] = labelFooWant
  289. n.Labels[v1.LabelZoneRegion] = "regionone"
  290. wantPrice := "0.133700"
  291. n2 := &v1.Node{}
  292. n2.Spec.ProviderID = providerIDWant
  293. n2.Name = nameWant
  294. n2.Labels = make(map[string]string)
  295. n2.Labels["foo"] = labelFooWant
  296. n2.Labels[v1.LabelZoneRegion] = "regiontwo"
  297. wantPrice2 := "0.133800"
  298. n3 := &v1.Node{}
  299. n3.Spec.ProviderID = providerIDWant
  300. n3.Name = nameWant
  301. n3.Labels = make(map[string]string)
  302. n3.Labels["foo"] = labelFooWant
  303. n3.Labels[v1.LabelZoneRegion] = "fakeregion"
  304. wantPrice3 := "0.1339"
  305. c := &provider.CSVProvider{
  306. CSVLocation: "../configs/pricing_schema_region.csv",
  307. CustomProvider: &provider.CustomProvider{
  308. Config: provider.NewProviderConfig(confMan, "../configs/default.json"),
  309. },
  310. }
  311. c.DownloadPricingData()
  312. k := c.GetKey(n.Labels, n)
  313. resN, _, err := c.NodePricing(k)
  314. if err != nil {
  315. t.Errorf("Error in NodePricing: %s", err.Error())
  316. } else {
  317. gotPrice := resN.Cost
  318. if gotPrice != wantPrice {
  319. t.Errorf("Wanted price '%s' got price '%s'", wantPrice, gotPrice)
  320. }
  321. }
  322. k2 := c.GetKey(n2.Labels, n2)
  323. resN2, _, err := c.NodePricing(k2)
  324. if err != nil {
  325. t.Errorf("Error in NodePricing: %s", err.Error())
  326. } else {
  327. gotPrice := resN2.Cost
  328. if gotPrice != wantPrice2 {
  329. t.Errorf("Wanted price '%s' got price '%s'", wantPrice2, gotPrice)
  330. }
  331. }
  332. k3 := c.GetKey(n3.Labels, n3)
  333. resN3, _, err := c.NodePricing(k3)
  334. if err != nil {
  335. t.Errorf("Error in NodePricing: %s", err.Error())
  336. } else {
  337. gotPrice := resN3.Cost
  338. if gotPrice != wantPrice3 {
  339. t.Errorf("Wanted price '%s' got price '%s'", wantPrice3, gotPrice)
  340. }
  341. }
  342. unknownN := &v1.Node{}
  343. unknownN.Spec.ProviderID = "fake providerID"
  344. unknownN.Name = "unknownname"
  345. unknownN.Labels = make(map[string]string)
  346. unknownN.Labels["topology.kubernetes.io/region"] = "fakeregion"
  347. unknownN.Labels["foo"] = labelFooWant
  348. k4 := c.GetKey(unknownN.Labels, unknownN)
  349. resN4, _, _ := c.NodePricing(k4)
  350. if resN4 != nil {
  351. t.Errorf("CSV provider should return nil on missing node, instead returned %+v", resN4)
  352. }
  353. c2 := &provider.CSVProvider{
  354. CSVLocation: "../configs/fake.csv",
  355. CustomProvider: &provider.CustomProvider{
  356. Config: provider.NewProviderConfig(confMan, "../configs/default.json"),
  357. },
  358. }
  359. k5 := c.GetKey(n.Labels, n)
  360. resN5, _, _ := c2.NodePricing(k5)
  361. if resN5 != nil {
  362. t.Errorf("CSV provider should return nil on missing csv")
  363. }
  364. }
  365. type FakeCache struct {
  366. nodes []*v1.Node
  367. clustercache.ClusterCache
  368. }
  369. func (f FakeCache) GetAllNodes() []*v1.Node {
  370. return f.nodes
  371. }
  372. func (f FakeCache) GetAllDaemonSets() []*appsv1.DaemonSet {
  373. return nil
  374. }
  375. func NewFakeNodeCache(nodes []*v1.Node) FakeCache {
  376. return FakeCache{
  377. nodes: nodes,
  378. }
  379. }
  380. type FakeClusterMap struct {
  381. clusters.ClusterMap
  382. }
  383. func TestNodePriceFromCSVWithBadConfig(t *testing.T) {
  384. os.Setenv("CONFIG_PATH", "../config")
  385. confMan := config.NewConfigFileManager(&config.ConfigFileManagerOpts{
  386. LocalConfigPath: "./",
  387. })
  388. c := &provider.CSVProvider{
  389. CSVLocation: "../configs/pricing_schema_case.csv",
  390. CustomProvider: &provider.CustomProvider{
  391. Config: provider.NewProviderConfig(confMan, "invalid.json"),
  392. },
  393. }
  394. c.DownloadPricingData()
  395. n := &v1.Node{}
  396. n.Spec.ProviderID = "fake"
  397. n.Name = "nameWant"
  398. n.Labels = make(map[string]string)
  399. n.Labels["foo"] = "labelFooWant"
  400. n.Labels[v1.LabelZoneRegion] = "regionone"
  401. fc := NewFakeNodeCache([]*v1.Node{n})
  402. fm := FakeClusterMap{}
  403. d, _ := time.ParseDuration("1m")
  404. model := costmodel.NewCostModel(nil, nil, fc, fm, d)
  405. _, err := model.GetNodeCost(c)
  406. if err != nil {
  407. t.Errorf("Error in node pricing: %s", err)
  408. }
  409. }
  410. func TestSourceMatchesFromCSV(t *testing.T) {
  411. os.Setenv("CONFIG_PATH", "../configs")
  412. confMan := config.NewConfigFileManager(&config.ConfigFileManagerOpts{
  413. LocalConfigPath: "./",
  414. })
  415. c := &provider.CSVProvider{
  416. CSVLocation: "../configs/pricing_schema_case.csv",
  417. CustomProvider: &provider.CustomProvider{
  418. Config: provider.NewProviderConfig(confMan, "/default.json"),
  419. },
  420. }
  421. c.DownloadPricingData()
  422. n := &v1.Node{}
  423. n.Spec.ProviderID = "fake"
  424. n.Name = "nameWant"
  425. n.Labels = make(map[string]string)
  426. n.Labels["foo"] = "labelFooWant"
  427. n.Labels[v1.LabelZoneRegion] = "regionone"
  428. n2 := &v1.Node{}
  429. 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"
  430. n2.Labels = make(map[string]string)
  431. n2.Labels[v1.LabelZoneRegion] = "eastus2"
  432. n2.Labels["foo"] = "labelFooWant"
  433. k := c.GetKey(n2.Labels, n2)
  434. resN, _, err := c.NodePricing(k)
  435. if err != nil {
  436. t.Errorf("Error in NodePricing: %s", err.Error())
  437. } else {
  438. wantPrice := "0.13370357"
  439. gotPrice := resN.Cost
  440. if gotPrice != wantPrice {
  441. t.Errorf("Wanted price '%s' got price '%s'", wantPrice, gotPrice)
  442. }
  443. }
  444. n3 := &v1.Node{}
  445. n3.Spec.ProviderID = "fake"
  446. n3.Name = "nameWant"
  447. n3.Labels = make(map[string]string)
  448. n.Labels[v1.LabelZoneRegion] = "eastus2"
  449. n.Labels[v1.LabelInstanceType] = "Standard_F32s_v2"
  450. fc := NewFakeNodeCache([]*v1.Node{n, n2, n3})
  451. fm := FakeClusterMap{}
  452. d, _ := time.ParseDuration("1m")
  453. model := costmodel.NewCostModel(nil, nil, fc, fm, d)
  454. _, err = model.GetNodeCost(c)
  455. if err != nil {
  456. t.Errorf("Error in node pricing: %s", err)
  457. }
  458. p, err := model.GetPricingSourceCounts()
  459. if err != nil {
  460. t.Errorf("Error in pricing source counts: %s", err)
  461. } else if p.TotalNodes != 3 {
  462. t.Errorf("Wanted 3 nodes got %d", p.TotalNodes)
  463. }
  464. if p.PricingTypeCounts[""] != 1 {
  465. t.Errorf("Wanted 1 default match got %d: %+v", p.PricingTypeCounts[""], p.PricingTypeCounts)
  466. }
  467. if p.PricingTypeCounts["csvExact"] != 1 {
  468. t.Errorf("Wanted 1 exact match got %d: %+v", p.PricingTypeCounts["csvExact"], p.PricingTypeCounts)
  469. }
  470. if p.PricingTypeCounts["csvClass"] != 1 {
  471. t.Errorf("Wanted 1 class match got %d: %+v", p.PricingTypeCounts["csvClass"], p.PricingTypeCounts)
  472. }
  473. }
  474. func TestNodePriceFromCSVWithCase(t *testing.T) {
  475. n := &v1.Node{}
  476. 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"
  477. n.Labels = make(map[string]string)
  478. n.Labels[v1.LabelZoneRegion] = "eastus2"
  479. wantPrice := "0.13370357"
  480. confMan := config.NewConfigFileManager(&config.ConfigFileManagerOpts{
  481. LocalConfigPath: "./",
  482. })
  483. c := &provider.CSVProvider{
  484. CSVLocation: "../configs/pricing_schema_case.csv",
  485. CustomProvider: &provider.CustomProvider{
  486. Config: provider.NewProviderConfig(confMan, "../configs/default.json"),
  487. },
  488. }
  489. c.DownloadPricingData()
  490. k := c.GetKey(n.Labels, n)
  491. resN, _, err := c.NodePricing(k)
  492. if err != nil {
  493. t.Errorf("Error in NodePricing: %s", err.Error())
  494. } else {
  495. gotPrice := resN.Cost
  496. if gotPrice != wantPrice {
  497. t.Errorf("Wanted price '%s' got price '%s'", wantPrice, gotPrice)
  498. }
  499. }
  500. }
  501. func TestNodePriceFromCSVByClass(t *testing.T) {
  502. n := &v1.Node{}
  503. n.Spec.ProviderID = "fakeproviderid"
  504. n.Labels = make(map[string]string)
  505. n.Labels[v1.LabelZoneRegion] = "eastus2"
  506. n.Labels[v1.LabelInstanceType] = "Standard_F32s_v2"
  507. wantpricefloat := 0.13370357
  508. wantPrice := fmt.Sprintf("%f", (math.Round(wantpricefloat*1000000) / 1000000))
  509. confMan := config.NewConfigFileManager(&config.ConfigFileManagerOpts{
  510. LocalConfigPath: "./",
  511. })
  512. c := &provider.CSVProvider{
  513. CSVLocation: "../configs/pricing_schema_case.csv",
  514. CustomProvider: &provider.CustomProvider{
  515. Config: provider.NewProviderConfig(confMan, "../configs/default.json"),
  516. },
  517. }
  518. c.DownloadPricingData()
  519. k := c.GetKey(n.Labels, n)
  520. resN, _, err := c.NodePricing(k)
  521. if err != nil {
  522. t.Errorf("Error in NodePricing: %s", err.Error())
  523. } else {
  524. gotPrice := resN.Cost
  525. if gotPrice != wantPrice {
  526. t.Errorf("Wanted price '%s' got price '%s'", wantPrice, gotPrice)
  527. }
  528. }
  529. n2 := &v1.Node{}
  530. n2.Spec.ProviderID = "fakeproviderid"
  531. n2.Labels = make(map[string]string)
  532. n2.Labels[v1.LabelZoneRegion] = "fakeregion"
  533. n2.Labels[v1.LabelInstanceType] = "Standard_F32s_v2"
  534. k2 := c.GetKey(n2.Labels, n)
  535. c.DownloadPricingData()
  536. resN2, _, err := c.NodePricing(k2)
  537. if resN2 != nil {
  538. t.Errorf("CSV provider should return nil on missing node, instead returned %+v", resN2)
  539. }
  540. }