metricsquerier_test.go 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053
  1. package collector
  2. import (
  3. "cmp"
  4. "reflect"
  5. "slices"
  6. "testing"
  7. "time"
  8. "github.com/opencost/opencost/core/pkg/source"
  9. "github.com/opencost/opencost/core/pkg/util"
  10. "github.com/opencost/opencost/modules/collector-source/pkg/metric"
  11. )
  12. var Start1Str = "2025-01-01T00:00:00Z"
  13. var End1Str = "2025-01-01T01:00:00Z"
  14. type MockStoreProvider struct {
  15. metricsCollector metric.MetricStore
  16. }
  17. func (m *MockStoreProvider) GetStore(start, end time.Time) metric.MetricStore {
  18. return m.metricsCollector
  19. }
  20. // QueryDataCoverage is not implemented for this mock
  21. func (m *MockStoreProvider) GetDailyDataCoverage(limitDays int) (time.Time, time.Time, error) {
  22. return time.Time{}, time.Time{}, nil
  23. }
  24. func GetMockCollectorProvider() StoreProvider {
  25. collector := NewOpenCostMetricStore()
  26. start, _ := time.Parse(time.RFC3339, Start1Str)
  27. time1 := time.Date(2025, 1, 1, 0, 30, 0, 0, time.UTC)
  28. end, _ := time.Parse(time.RFC3339, End1Str)
  29. node1Info := map[string]string{
  30. "node": "node1",
  31. "provider_id": "node1",
  32. }
  33. localStorage1Info := map[string]string{
  34. source.InstanceLabel: "node1",
  35. source.DeviceLabel: "local",
  36. }
  37. cluster1Info := map[string]string{
  38. "provisioner_name": "GKE",
  39. }
  40. gpu1Info := map[string]string{
  41. source.NamespaceLabel: "namespace1",
  42. source.PodLabel: "pod1",
  43. source.UIDLabel: "pod-uuid1",
  44. "container": "container1",
  45. "gpu": "0",
  46. "UUID": "GPU-1",
  47. "pci_bus_id": "00000000:00:0A.0",
  48. "device": "nvidia0",
  49. "modelName": "Tesla T4",
  50. "Hostname": "localhost",
  51. }
  52. pod1Info := map[string]string{
  53. source.NamespaceLabel: "namespace1",
  54. source.NodeLabel: "node1",
  55. source.InstanceLabel: "node1",
  56. source.PodLabel: "pod1",
  57. source.UIDLabel: "pod-uuid1",
  58. }
  59. container1Info := map[string]string{
  60. source.NamespaceLabel: "namespace1",
  61. source.NodeLabel: "node1",
  62. source.InstanceLabel: "node1",
  63. source.PodLabel: "pod1",
  64. source.UIDLabel: "pod-uuid1",
  65. source.ContainerLabel: "container1",
  66. }
  67. container2Info := map[string]string{
  68. source.NamespaceLabel: "kube-system",
  69. source.NodeLabel: "node1",
  70. source.InstanceLabel: "node1",
  71. source.PodLabel: "pod2",
  72. source.UIDLabel: "pod-uuid2",
  73. source.ContainerLabel: "container2",
  74. }
  75. networkZone1Info := map[string]string{
  76. source.PodNameLabel: "pod1",
  77. source.NamespaceLabel: "namespace1",
  78. source.InternetLabel: "false",
  79. source.SameRegionLabel: "true",
  80. source.SameZoneLabel: "false",
  81. source.ServiceLabel: "service1",
  82. }
  83. networkRegion1Info := map[string]string{
  84. source.PodNameLabel: "pod1",
  85. source.NamespaceLabel: "namespace1",
  86. source.InternetLabel: "false",
  87. source.SameRegionLabel: "false",
  88. source.SameZoneLabel: "false",
  89. source.ServiceLabel: "service1",
  90. }
  91. networkInternet1Info := map[string]string{
  92. source.PodNameLabel: "pod1",
  93. source.NamespaceLabel: "namespace1",
  94. source.InternetLabel: "true",
  95. source.SameRegionLabel: "false",
  96. source.SameZoneLabel: "false",
  97. source.ServiceLabel: "service1",
  98. }
  99. networkInternet2Info := map[string]string{
  100. source.PodNameLabel: "pod1",
  101. source.NamespaceLabel: "namespace1",
  102. source.InternetLabel: "true",
  103. source.SameRegionLabel: "false",
  104. source.SameZoneLabel: "false",
  105. source.ServiceLabel: "service2",
  106. }
  107. collector.Update(metric.NodeTotalHourlyCost, node1Info, 0, start, nil)
  108. collector.Update(metric.NodeTotalHourlyCost, node1Info, 0, end, nil)
  109. collector.Update(metric.NodeFSCapacityBytes, localStorage1Info, 2*GiB, start, nil)
  110. collector.Update(metric.ContainerFSUsageBytes, localStorage1Info, 1*GiB, start, nil)
  111. collector.Update(metric.ContainerFSUsageBytes, localStorage1Info, 1*GiB, end, nil)
  112. collector.Update(metric.KubeNodeStatusCapacityMemoryBytes, node1Info, 4*GiB, start, nil)
  113. collector.Update(metric.ContainerMemoryWorkingSetBytes, container1Info, 1*GiB, start, nil)
  114. collector.Update(metric.ContainerMemoryWorkingSetBytes, container2Info, 2*GiB, start, nil)
  115. collector.Update(metric.ContainerCPUUsageSecondsTotal, container1Info, 0, start, nil)
  116. collector.Update(metric.ContainerCPUUsageSecondsTotal, container1Info, 60*60*4, time1, nil)
  117. collector.Update(metric.ContainerCPUUsageSecondsTotal, container1Info, 60*60*10, end, nil)
  118. collector.Update(metric.KubecostClusterManagementCost, cluster1Info, 0.1, start, nil)
  119. collector.Update(metric.KubecostClusterManagementCost, cluster1Info, 0.1, end, nil)
  120. collector.Update(metric.DCGMFIDEVDECUTIL, gpu1Info, 0, start, nil)
  121. collector.Update(metric.DCGMFIPROFGRENGINEACTIVE, gpu1Info, 0, start, nil)
  122. collector.Update(metric.DCGMFIPROFGRENGINEACTIVE, gpu1Info, 1, end, nil)
  123. collector.Update(metric.KubecostNetworkZoneEgressCost, nil, 1, start, nil)
  124. collector.Update(metric.KubecostNetworkRegionEgressCost, nil, 2, start, nil)
  125. collector.Update(metric.KubecostNetworkInternetEgressCost, nil, 3, start, nil)
  126. collector.Update(metric.ContainerNetworkTransmitBytesTotal, pod1Info, 3*GiB, start, nil)
  127. collector.Update(metric.ContainerNetworkTransmitBytesTotal, pod1Info, 13*GiB, end, nil)
  128. collector.Update(metric.ContainerNetworkReceiveBytesTotal, pod1Info, 30*GiB, start, nil)
  129. collector.Update(metric.ContainerNetworkReceiveBytesTotal, pod1Info, 130*GiB, end, nil)
  130. collector.Update(metric.KubecostPodNetworkEgressBytesTotal, networkRegion1Info, 1*GiB, start, nil)
  131. collector.Update(metric.KubecostPodNetworkEgressBytesTotal, networkZone1Info, 0*GiB, start, nil)
  132. collector.Update(metric.KubecostPodNetworkEgressBytesTotal, networkInternet1Info, 1*GiB, start, nil)
  133. collector.Update(metric.KubecostPodNetworkEgressBytesTotal, networkInternet2Info, 1*GiB, start, nil)
  134. collector.Update(metric.KubecostPodNetworkEgressBytesTotal, networkRegion1Info, 2*GiB, end, nil)
  135. collector.Update(metric.KubecostPodNetworkEgressBytesTotal, networkZone1Info, 2*GiB, end, nil)
  136. collector.Update(metric.KubecostPodNetworkEgressBytesTotal, networkInternet1Info, 4*GiB, end, nil)
  137. collector.Update(metric.KubecostPodNetworkEgressBytesTotal, networkInternet2Info, 5*GiB, end, nil)
  138. collector.Update(metric.KubecostPodNetworkIngressBytesTotal, networkRegion1Info, 10*GiB, start, nil)
  139. collector.Update(metric.KubecostPodNetworkIngressBytesTotal, networkZone1Info, 0*GiB, start, nil)
  140. collector.Update(metric.KubecostPodNetworkIngressBytesTotal, networkInternet1Info, 10*GiB, start, nil)
  141. collector.Update(metric.KubecostPodNetworkIngressBytesTotal, networkInternet2Info, 10*GiB, start, nil)
  142. collector.Update(metric.KubecostPodNetworkIngressBytesTotal, networkRegion1Info, 20*GiB, end, nil)
  143. collector.Update(metric.KubecostPodNetworkIngressBytesTotal, networkZone1Info, 20*GiB, end, nil)
  144. collector.Update(metric.KubecostPodNetworkIngressBytesTotal, networkInternet1Info, 40*GiB, end, nil)
  145. collector.Update(metric.KubecostPodNetworkIngressBytesTotal, networkInternet2Info, 50*GiB, end, nil)
  146. return &MockStoreProvider{
  147. metricsCollector: collector,
  148. }
  149. }
  150. func TestCollectorMetricsQuerier_QueryLocalStorageCost(t *testing.T) {
  151. start1, _ := time.Parse(time.RFC3339, Start1Str)
  152. end1, _ := time.Parse(time.RFC3339, End1Str)
  153. c := collectorMetricsQuerier{
  154. collectorProvider: GetMockCollectorProvider(),
  155. }
  156. resCh := c.QueryLocalStorageCost(start1, end1)
  157. res, err := resCh.Await()
  158. if err != nil {
  159. t.Errorf("unexpected error: %v", err.Error())
  160. }
  161. expected := []*source.LocalStorageCostResult{
  162. {
  163. Cluster: "",
  164. Instance: "node1",
  165. Device: "local",
  166. Data: []*util.Vector{
  167. {
  168. Value: LocalStorageCostPerGiBHr * 2,
  169. },
  170. },
  171. },
  172. }
  173. if len(res) != len(expected) {
  174. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  175. }
  176. for i, got := range res {
  177. if !reflect.DeepEqual(got, expected[i]) {
  178. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  179. }
  180. }
  181. }
  182. func TestCollectorMetricsQuerier_QueryLocalStorageUsedCost(t *testing.T) {
  183. start1, _ := time.Parse(time.RFC3339, Start1Str)
  184. end1, _ := time.Parse(time.RFC3339, End1Str)
  185. c := collectorMetricsQuerier{
  186. collectorProvider: GetMockCollectorProvider(),
  187. }
  188. resCh := c.QueryLocalStorageUsedCost(start1, end1)
  189. res, err := resCh.Await()
  190. if err != nil {
  191. t.Errorf("unexpected error: %v", err.Error())
  192. }
  193. expected := []*source.LocalStorageUsedCostResult{
  194. {
  195. Cluster: "",
  196. Instance: "node1",
  197. Device: "local",
  198. Data: []*util.Vector{
  199. {
  200. Value: LocalStorageCostPerGiBHr,
  201. },
  202. },
  203. },
  204. }
  205. if len(res) != len(expected) {
  206. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  207. }
  208. for i, got := range res {
  209. if !reflect.DeepEqual(got, expected[i]) {
  210. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  211. }
  212. }
  213. }
  214. func TestCollectorMetricsQuerier_QueryNodeActiveMinutes(t *testing.T) {
  215. start1, _ := time.Parse(time.RFC3339, Start1Str)
  216. end1, _ := time.Parse(time.RFC3339, End1Str)
  217. c := collectorMetricsQuerier{
  218. collectorProvider: GetMockCollectorProvider(),
  219. }
  220. resCh := c.QueryNodeActiveMinutes(time.Now(), time.Now())
  221. res, err := resCh.Await()
  222. if err != nil {
  223. t.Errorf("unexpected error: %v", err.Error())
  224. }
  225. expected := []*source.NodeActiveMinutesResult{
  226. {
  227. Cluster: "",
  228. Node: "node1",
  229. ProviderID: "node1",
  230. Data: []*util.Vector{
  231. {
  232. Timestamp: float64(start1.Unix()),
  233. Value: 1,
  234. },
  235. {
  236. Timestamp: float64(end1.Unix()),
  237. Value: 1,
  238. },
  239. },
  240. },
  241. }
  242. if len(res) != len(expected) {
  243. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  244. }
  245. for i, got := range res {
  246. if !reflect.DeepEqual(got, expected[i]) {
  247. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  248. }
  249. }
  250. }
  251. func TestCollectorMetricsQuerier_QueryNodeRAMSystemPercent(t *testing.T) {
  252. start1, _ := time.Parse(time.RFC3339, Start1Str)
  253. end1, _ := time.Parse(time.RFC3339, End1Str)
  254. c := collectorMetricsQuerier{
  255. collectorProvider: GetMockCollectorProvider(),
  256. }
  257. resCh := c.QueryNodeRAMSystemPercent(start1, end1)
  258. res, err := resCh.Await()
  259. if err != nil {
  260. t.Errorf("unexpected error: %v", err.Error())
  261. }
  262. expected := []*source.NodeRAMSystemPercentResult{
  263. {
  264. UID: "pod-uuid2",
  265. Cluster: "",
  266. Instance: "node1",
  267. Data: []*util.Vector{
  268. {
  269. Value: .5,
  270. },
  271. },
  272. },
  273. }
  274. if len(res) != len(expected) {
  275. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  276. }
  277. for i, got := range res {
  278. if !reflect.DeepEqual(got, expected[i]) {
  279. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  280. }
  281. }
  282. }
  283. func TestCollectorMetricsQuerier_QueryNodeRAMUserPercent(t *testing.T) {
  284. start1, _ := time.Parse(time.RFC3339, Start1Str)
  285. end1, _ := time.Parse(time.RFC3339, End1Str)
  286. c := collectorMetricsQuerier{
  287. collectorProvider: GetMockCollectorProvider(),
  288. }
  289. resCh := c.QueryNodeRAMUserPercent(start1, end1)
  290. res, err := resCh.Await()
  291. if err != nil {
  292. t.Errorf("unexpected error: %v", err.Error())
  293. }
  294. expected := []*source.NodeRAMUserPercentResult{
  295. {
  296. UID: "pod-uuid1",
  297. Cluster: "",
  298. Instance: "node1",
  299. Data: []*util.Vector{
  300. {
  301. Value: .25,
  302. },
  303. },
  304. },
  305. }
  306. if len(res) != len(expected) {
  307. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  308. }
  309. for i, got := range res {
  310. if !reflect.DeepEqual(got, expected[i]) {
  311. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  312. }
  313. }
  314. }
  315. func TestCollectorMetricsQuerier_QueryClusterManagementDuration(t *testing.T) {
  316. start1, _ := time.Parse(time.RFC3339, Start1Str)
  317. end1, _ := time.Parse(time.RFC3339, End1Str)
  318. c := collectorMetricsQuerier{
  319. collectorProvider: GetMockCollectorProvider(),
  320. }
  321. resCh := c.QueryClusterManagementDuration(start1, end1)
  322. res, err := resCh.Await()
  323. if err != nil {
  324. t.Errorf("unexpected error: %v", err.Error())
  325. }
  326. expected := []*source.ClusterManagementDurationResult{
  327. {
  328. Cluster: "",
  329. Provisioner: "GKE",
  330. Data: []*util.Vector{
  331. {
  332. Timestamp: float64(start1.Unix()),
  333. Value: 1,
  334. },
  335. {
  336. Timestamp: float64(end1.Unix()),
  337. Value: 1,
  338. },
  339. },
  340. },
  341. }
  342. if len(res) != len(expected) {
  343. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  344. }
  345. for i, got := range res {
  346. if !reflect.DeepEqual(got, expected[i]) {
  347. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  348. }
  349. }
  350. }
  351. func Test_collectorMetricsQuerier_QueryCPUUsageAvg(t *testing.T) {
  352. start1, _ := time.Parse(time.RFC3339, Start1Str)
  353. end1, _ := time.Parse(time.RFC3339, End1Str)
  354. c := collectorMetricsQuerier{
  355. collectorProvider: GetMockCollectorProvider(),
  356. }
  357. resCh := c.QueryCPUUsageAvg(start1, end1)
  358. res, err := resCh.Await()
  359. if err != nil {
  360. t.Errorf("unexpected error: %v", err.Error())
  361. }
  362. expected := []*source.CPUUsageAvgResult{
  363. {
  364. UID: "pod-uuid1",
  365. Cluster: "",
  366. Namespace: "namespace1",
  367. Node: "node1",
  368. Instance: "node1",
  369. Pod: "pod1",
  370. Container: "container1",
  371. Data: []*util.Vector{
  372. {
  373. Value: 10,
  374. },
  375. },
  376. },
  377. }
  378. if len(res) != len(expected) {
  379. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  380. }
  381. for i, got := range res {
  382. if !reflect.DeepEqual(got, expected[i]) {
  383. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  384. }
  385. }
  386. }
  387. func Test_collectorMetricsQuerier_QueryCPUUsageMax(t *testing.T) {
  388. start1, _ := time.Parse(time.RFC3339, Start1Str)
  389. end1, _ := time.Parse(time.RFC3339, End1Str)
  390. c := collectorMetricsQuerier{
  391. collectorProvider: GetMockCollectorProvider(),
  392. }
  393. resCh := c.QueryCPUUsageMax(start1, end1)
  394. res, err := resCh.Await()
  395. if err != nil {
  396. t.Errorf("unexpected error: %v", err.Error())
  397. }
  398. expected := []*source.CPUUsageMaxResult{
  399. {
  400. UID: "pod-uuid1",
  401. Cluster: "",
  402. Namespace: "namespace1",
  403. Node: "node1",
  404. Instance: "node1",
  405. Pod: "pod1",
  406. Container: "container1",
  407. Data: []*util.Vector{
  408. {
  409. Value: 12,
  410. },
  411. },
  412. },
  413. }
  414. if len(res) != len(expected) {
  415. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  416. }
  417. for i, got := range res {
  418. if !reflect.DeepEqual(got, expected[i]) {
  419. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  420. }
  421. }
  422. }
  423. func TestCollectorMetricsQuerier_QueryGPUsUsageAvg(t *testing.T) {
  424. start1, _ := time.Parse(time.RFC3339, Start1Str)
  425. end1, _ := time.Parse(time.RFC3339, End1Str)
  426. c := collectorMetricsQuerier{
  427. collectorProvider: GetMockCollectorProvider(),
  428. }
  429. resCh := c.QueryGPUsUsageAvg(start1, end1)
  430. res, err := resCh.Await()
  431. if err != nil {
  432. t.Errorf("unexpected error: %v", err.Error())
  433. }
  434. expected := []*source.GPUsUsageAvgResult{
  435. {
  436. UID: "pod-uuid1",
  437. Cluster: "",
  438. Namespace: "namespace1",
  439. Pod: "pod1",
  440. Container: "container1",
  441. Data: []*util.Vector{
  442. {
  443. Value: 0.5,
  444. },
  445. },
  446. },
  447. }
  448. if len(res) != len(expected) {
  449. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  450. }
  451. for i, got := range res {
  452. if !reflect.DeepEqual(got, expected[i]) {
  453. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  454. }
  455. }
  456. }
  457. func TestCollectorMetricsQuerier_QueryGPUsUsageMax(t *testing.T) {
  458. start1, _ := time.Parse(time.RFC3339, Start1Str)
  459. end1, _ := time.Parse(time.RFC3339, End1Str)
  460. c := collectorMetricsQuerier{
  461. collectorProvider: GetMockCollectorProvider(),
  462. }
  463. resCh := c.QueryGPUsUsageMax(start1, end1)
  464. res, err := resCh.Await()
  465. if err != nil {
  466. t.Errorf("unexpected error: %v", err.Error())
  467. }
  468. expected := []*source.GPUsUsageMaxResult{
  469. {
  470. UID: "pod-uuid1",
  471. Cluster: "",
  472. Namespace: "namespace1",
  473. Pod: "pod1",
  474. Container: "container1",
  475. Data: []*util.Vector{
  476. {
  477. Value: 1.0,
  478. },
  479. },
  480. },
  481. }
  482. if len(res) != len(expected) {
  483. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  484. }
  485. for i, got := range res {
  486. if !reflect.DeepEqual(got, expected[i]) {
  487. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  488. }
  489. }
  490. }
  491. func TestCollectorMetricsQuerier_QueryGPUInfo(t *testing.T) {
  492. start1, _ := time.Parse(time.RFC3339, Start1Str)
  493. end1, _ := time.Parse(time.RFC3339, End1Str)
  494. c := collectorMetricsQuerier{
  495. collectorProvider: GetMockCollectorProvider(),
  496. }
  497. resCh := c.QueryGPUInfo(start1, end1)
  498. res, err := resCh.Await()
  499. if err != nil {
  500. t.Errorf("unexpected error: %v", err.Error())
  501. }
  502. expected := []*source.GPUInfoResult{
  503. {
  504. UID: "pod-uuid1",
  505. Cluster: "",
  506. Namespace: "namespace1",
  507. Pod: "pod1",
  508. Container: "container1",
  509. Device: "nvidia0",
  510. ModelName: "Tesla T4",
  511. UUID: "GPU-1",
  512. Data: []*util.Vector{
  513. {
  514. Value: 1,
  515. },
  516. },
  517. },
  518. }
  519. if len(res) != len(expected) {
  520. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  521. }
  522. for i, got := range res {
  523. if !reflect.DeepEqual(got, expected[i]) {
  524. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  525. }
  526. }
  527. }
  528. func Test_collectorMetricsQuerier_QueryNetZoneGiB(t *testing.T) {
  529. start1, _ := time.Parse(time.RFC3339, Start1Str)
  530. end1, _ := time.Parse(time.RFC3339, End1Str)
  531. c := collectorMetricsQuerier{
  532. collectorProvider: GetMockCollectorProvider(),
  533. }
  534. resCh := c.QueryNetZoneGiB(start1, end1)
  535. res, err := resCh.Await()
  536. if err != nil {
  537. t.Errorf("unexpected error: %v", err.Error())
  538. }
  539. expected := []*source.NetZoneGiBResult{
  540. {
  541. Cluster: "",
  542. Namespace: "namespace1",
  543. Pod: "pod1",
  544. Data: []*util.Vector{
  545. {
  546. Value: 2,
  547. },
  548. },
  549. },
  550. }
  551. if len(res) != len(expected) {
  552. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  553. }
  554. for i, got := range res {
  555. if !reflect.DeepEqual(got, expected[i]) {
  556. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  557. }
  558. }
  559. }
  560. func Test_collectorMetricsQuerier_QueryNetZonePricePerGiB(t *testing.T) {
  561. start1, _ := time.Parse(time.RFC3339, Start1Str)
  562. end1, _ := time.Parse(time.RFC3339, End1Str)
  563. c := collectorMetricsQuerier{
  564. collectorProvider: GetMockCollectorProvider(),
  565. }
  566. resCh := c.QueryNetZonePricePerGiB(start1, end1)
  567. res, err := resCh.Await()
  568. if err != nil {
  569. t.Errorf("unexpected error: %v", err.Error())
  570. }
  571. expected := []*source.NetZonePricePerGiBResult{
  572. {
  573. Cluster: "",
  574. Data: []*util.Vector{
  575. {
  576. Value: 1,
  577. },
  578. },
  579. },
  580. }
  581. if len(res) != len(expected) {
  582. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  583. }
  584. for i, got := range res {
  585. if !reflect.DeepEqual(got, expected[i]) {
  586. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  587. }
  588. }
  589. }
  590. func Test_collectorMetricsQuerier_QueryNetRegionGiB(t *testing.T) {
  591. start1, _ := time.Parse(time.RFC3339, Start1Str)
  592. end1, _ := time.Parse(time.RFC3339, End1Str)
  593. c := collectorMetricsQuerier{
  594. collectorProvider: GetMockCollectorProvider(),
  595. }
  596. resCh := c.QueryNetRegionGiB(start1, end1)
  597. res, err := resCh.Await()
  598. if err != nil {
  599. t.Errorf("unexpected error: %v", err.Error())
  600. }
  601. expected := []*source.NetRegionGiBResult{
  602. {
  603. Cluster: "",
  604. Namespace: "namespace1",
  605. Pod: "pod1",
  606. Data: []*util.Vector{
  607. {
  608. Value: 1,
  609. },
  610. },
  611. },
  612. }
  613. if len(res) != len(expected) {
  614. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  615. }
  616. for i, got := range res {
  617. if !reflect.DeepEqual(got, expected[i]) {
  618. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  619. }
  620. }
  621. }
  622. func Test_collectorMetricsQuerier_QueryNetRegionPricePerGiB(t *testing.T) {
  623. start1, _ := time.Parse(time.RFC3339, Start1Str)
  624. end1, _ := time.Parse(time.RFC3339, End1Str)
  625. c := collectorMetricsQuerier{
  626. collectorProvider: GetMockCollectorProvider(),
  627. }
  628. resCh := c.QueryNetRegionPricePerGiB(start1, end1)
  629. res, err := resCh.Await()
  630. if err != nil {
  631. t.Errorf("unexpected error: %v", err.Error())
  632. }
  633. expected := []*source.NetRegionPricePerGiBResult{
  634. {
  635. Cluster: "",
  636. Data: []*util.Vector{
  637. {
  638. Value: 2,
  639. },
  640. },
  641. },
  642. }
  643. if len(res) != len(expected) {
  644. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  645. }
  646. for i, got := range res {
  647. if !reflect.DeepEqual(got, expected[i]) {
  648. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  649. }
  650. }
  651. }
  652. func Test_collectorMetricsQuerier_QueryNetInternetGiB(t *testing.T) {
  653. start1, _ := time.Parse(time.RFC3339, Start1Str)
  654. end1, _ := time.Parse(time.RFC3339, End1Str)
  655. c := collectorMetricsQuerier{
  656. collectorProvider: GetMockCollectorProvider(),
  657. }
  658. resCh := c.QueryNetInternetGiB(start1, end1)
  659. res, err := resCh.Await()
  660. if err != nil {
  661. t.Errorf("unexpected error: %v", err.Error())
  662. }
  663. expected := []*source.NetInternetGiBResult{
  664. {
  665. Cluster: "",
  666. Namespace: "namespace1",
  667. Pod: "pod1",
  668. Data: []*util.Vector{
  669. {
  670. Value: 7,
  671. },
  672. },
  673. },
  674. }
  675. if len(res) != len(expected) {
  676. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  677. }
  678. for i, got := range res {
  679. if !reflect.DeepEqual(got, expected[i]) {
  680. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  681. }
  682. }
  683. }
  684. func Test_collectorMetricsQuerier_QueryNetInternetPricePerGiB(t *testing.T) {
  685. start1, _ := time.Parse(time.RFC3339, Start1Str)
  686. end1, _ := time.Parse(time.RFC3339, End1Str)
  687. c := collectorMetricsQuerier{
  688. collectorProvider: GetMockCollectorProvider(),
  689. }
  690. resCh := c.QueryNetInternetPricePerGiB(start1, end1)
  691. res, err := resCh.Await()
  692. if err != nil {
  693. t.Errorf("unexpected error: %v", err.Error())
  694. }
  695. expected := []*source.NetInternetPricePerGiBResult{
  696. {
  697. Cluster: "",
  698. Data: []*util.Vector{
  699. {
  700. Value: 3,
  701. },
  702. },
  703. },
  704. }
  705. if len(res) != len(expected) {
  706. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  707. }
  708. for i, got := range res {
  709. if !reflect.DeepEqual(got, expected[i]) {
  710. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  711. }
  712. }
  713. }
  714. func Test_collectorMetricsQuerier_QueryNetInternetServiceGiB(t *testing.T) {
  715. start1, _ := time.Parse(time.RFC3339, Start1Str)
  716. end1, _ := time.Parse(time.RFC3339, End1Str)
  717. c := collectorMetricsQuerier{
  718. collectorProvider: GetMockCollectorProvider(),
  719. }
  720. resCh := c.QueryNetInternetServiceGiB(start1, end1)
  721. res, err := resCh.Await()
  722. if err != nil {
  723. t.Errorf("unexpected error: %v", err.Error())
  724. }
  725. expected := []*source.NetInternetServiceGiBResult{
  726. {
  727. Cluster: "",
  728. Namespace: "namespace1",
  729. Pod: "pod1",
  730. Service: "service1",
  731. Data: []*util.Vector{
  732. {
  733. Value: 3,
  734. },
  735. },
  736. },
  737. {
  738. Cluster: "",
  739. Namespace: "namespace1",
  740. Pod: "pod1",
  741. Service: "service2",
  742. Data: []*util.Vector{
  743. {
  744. Value: 4,
  745. },
  746. },
  747. },
  748. }
  749. if len(res) != len(expected) {
  750. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  751. }
  752. slices.SortFunc(res, func(a, b *source.NetInternetServiceGiBResult) int {
  753. return cmp.Compare(a.Service, b.Service)
  754. })
  755. for i, got := range res {
  756. if !reflect.DeepEqual(got, expected[i]) {
  757. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  758. }
  759. }
  760. }
  761. func Test_collectorMetricsQuerier_QueryNetTransferBytes(t *testing.T) {
  762. start1, _ := time.Parse(time.RFC3339, Start1Str)
  763. end1, _ := time.Parse(time.RFC3339, End1Str)
  764. c := collectorMetricsQuerier{
  765. collectorProvider: GetMockCollectorProvider(),
  766. }
  767. resCh := c.QueryNetTransferBytes(start1, end1)
  768. res, err := resCh.Await()
  769. if err != nil {
  770. t.Errorf("unexpected error: %v", err.Error())
  771. }
  772. expected := []*source.NetTransferBytesResult{
  773. {
  774. UID: "pod-uuid1",
  775. Cluster: "",
  776. Namespace: "namespace1",
  777. Pod: "pod1",
  778. Data: []*util.Vector{
  779. {
  780. Value: 10 * GiB,
  781. },
  782. },
  783. },
  784. }
  785. if len(res) != len(expected) {
  786. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  787. }
  788. for i, got := range res {
  789. if !reflect.DeepEqual(got, expected[i]) {
  790. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  791. }
  792. }
  793. }
  794. func Test_collectorMetricsQuerier_QueryNetZoneIngressGiB(t *testing.T) {
  795. start1, _ := time.Parse(time.RFC3339, Start1Str)
  796. end1, _ := time.Parse(time.RFC3339, End1Str)
  797. c := collectorMetricsQuerier{
  798. collectorProvider: GetMockCollectorProvider(),
  799. }
  800. resCh := c.QueryNetZoneIngressGiB(start1, end1)
  801. res, err := resCh.Await()
  802. if err != nil {
  803. t.Errorf("unexpected error: %v", err.Error())
  804. }
  805. expected := []*source.NetZoneIngressGiBResult{
  806. {
  807. Cluster: "",
  808. Namespace: "namespace1",
  809. Pod: "pod1",
  810. Data: []*util.Vector{
  811. {
  812. Value: 20,
  813. },
  814. },
  815. },
  816. }
  817. if len(res) != len(expected) {
  818. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  819. }
  820. for i, got := range res {
  821. if !reflect.DeepEqual(got, expected[i]) {
  822. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  823. }
  824. }
  825. }
  826. func Test_collectorMetricsQuerier_QueryNetRegionIngressGiB(t *testing.T) {
  827. start1, _ := time.Parse(time.RFC3339, Start1Str)
  828. end1, _ := time.Parse(time.RFC3339, End1Str)
  829. c := collectorMetricsQuerier{
  830. collectorProvider: GetMockCollectorProvider(),
  831. }
  832. resCh := c.QueryNetRegionIngressGiB(start1, end1)
  833. res, err := resCh.Await()
  834. if err != nil {
  835. t.Errorf("unexpected error: %v", err.Error())
  836. }
  837. expected := []*source.NetRegionIngressGiBResult{
  838. {
  839. Cluster: "",
  840. Namespace: "namespace1",
  841. Pod: "pod1",
  842. Data: []*util.Vector{
  843. {
  844. Value: 10,
  845. },
  846. },
  847. },
  848. }
  849. if len(res) != len(expected) {
  850. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  851. }
  852. for i, got := range res {
  853. if !reflect.DeepEqual(got, expected[i]) {
  854. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  855. }
  856. }
  857. }
  858. func Test_collectorMetricsQuerier_QueryNetInternetIngressGiB(t *testing.T) {
  859. start1, _ := time.Parse(time.RFC3339, Start1Str)
  860. end1, _ := time.Parse(time.RFC3339, End1Str)
  861. c := collectorMetricsQuerier{
  862. collectorProvider: GetMockCollectorProvider(),
  863. }
  864. resCh := c.QueryNetInternetIngressGiB(start1, end1)
  865. res, err := resCh.Await()
  866. if err != nil {
  867. t.Errorf("unexpected error: %v", err.Error())
  868. }
  869. expected := []*source.NetInternetIngressGiBResult{
  870. {
  871. Cluster: "",
  872. Namespace: "namespace1",
  873. Pod: "pod1",
  874. Data: []*util.Vector{
  875. {
  876. Value: 70,
  877. },
  878. },
  879. },
  880. }
  881. if len(res) != len(expected) {
  882. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  883. }
  884. for i, got := range res {
  885. if !reflect.DeepEqual(got, expected[i]) {
  886. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  887. }
  888. }
  889. }
  890. func Test_collectorMetricsQuerier_QueryNetInternetServiceIngressGiB(t *testing.T) {
  891. start1, _ := time.Parse(time.RFC3339, Start1Str)
  892. end1, _ := time.Parse(time.RFC3339, End1Str)
  893. c := collectorMetricsQuerier{
  894. collectorProvider: GetMockCollectorProvider(),
  895. }
  896. resCh := c.QueryNetInternetServiceIngressGiB(start1, end1)
  897. res, err := resCh.Await()
  898. if err != nil {
  899. t.Errorf("unexpected error: %v", err.Error())
  900. }
  901. expected := []*source.NetInternetServiceIngressGiBResult{
  902. {
  903. Cluster: "",
  904. Namespace: "namespace1",
  905. Pod: "pod1",
  906. Service: "service1",
  907. Data: []*util.Vector{
  908. {
  909. Value: 30,
  910. },
  911. },
  912. },
  913. {
  914. Cluster: "",
  915. Namespace: "namespace1",
  916. Pod: "pod1",
  917. Service: "service2",
  918. Data: []*util.Vector{
  919. {
  920. Value: 40,
  921. },
  922. },
  923. },
  924. }
  925. if len(res) != len(expected) {
  926. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  927. }
  928. slices.SortFunc(res, func(a, b *source.NetInternetServiceIngressGiBResult) int {
  929. return cmp.Compare(a.Service, b.Service)
  930. })
  931. for i, got := range res {
  932. if !reflect.DeepEqual(got, expected[i]) {
  933. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  934. }
  935. }
  936. }
  937. func Test_collectorMetricsQuerier_QueryNetReceiveBytes(t *testing.T) {
  938. start1, _ := time.Parse(time.RFC3339, Start1Str)
  939. end1, _ := time.Parse(time.RFC3339, End1Str)
  940. c := collectorMetricsQuerier{
  941. collectorProvider: GetMockCollectorProvider(),
  942. }
  943. resCh := c.QueryNetReceiveBytes(start1, end1)
  944. res, err := resCh.Await()
  945. if err != nil {
  946. t.Errorf("unexpected error: %v", err.Error())
  947. }
  948. expected := []*source.NetReceiveBytesResult{
  949. {
  950. UID: "pod-uuid1",
  951. Cluster: "",
  952. Namespace: "namespace1",
  953. Pod: "pod1",
  954. Data: []*util.Vector{
  955. {
  956. Value: 100 * GiB,
  957. },
  958. },
  959. },
  960. }
  961. if len(res) != len(expected) {
  962. t.Errorf("length of result was not as expected: got = %d, want %d", len(res), len(expected))
  963. }
  964. for i, got := range res {
  965. if !reflect.DeepEqual(got, expected[i]) {
  966. t.Errorf("result at index %d did not match: got = %v, want %v", i, got, expected[i])
  967. }
  968. }
  969. }