summaryrefslogtreecommitdiff
path: root/robusta_krr/formatters/csv.py
blob: 792c061f1f399ba9f5a212c47b22dd8b6a65935d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
import itertools
import csv

import logging


from robusta_krr.core.abstract import formatters
from robusta_krr.core.models.allocations import RecommendationValue
from robusta_krr.core.models.result import ResourceScan, ResourceType, Result
from robusta_krr.utils import resource_units
import datetime

logger = logging.getLogger("krr")

NONE_LITERAL = "unset"
NAN_LITERAL = "?"


def _format(value: RecommendationValue) -> str:
    if value is None:
        return NONE_LITERAL
    elif isinstance(value, str):
        return NAN_LITERAL
    else:
        return resource_units.format(value)


def __calc_diff(allocated, recommended, selector, multiplier=1) -> str:
    if recommended is None or isinstance(recommended.value, str) or selector != "requests":
        return ""
    else:
        reccomended_val = recommended.value if isinstance(recommended.value, (int, float)) else 0
        allocated_val = allocated if isinstance(allocated, (int, float)) else 0
        diff_val = reccomended_val - allocated_val
        diff_sign = "+" if diff_val >= 0 else "-"
        return f"{diff_sign}{_format(abs(diff_val) * multiplier)}"


def _format_request_str(item: ResourceScan, resource: ResourceType, selector: str) -> str:
    allocated = getattr(item.object.allocations, selector)[resource]
    recommended = getattr(item.recommended, selector)[resource]

    if allocated is None and recommended.value is None:
        return f"{NONE_LITERAL}"

    diff = __calc_diff(allocated, recommended, selector)
    if diff != "":
        diff = f"({diff}) "

    return (
        diff
        + _format(allocated)
        + " -> "
        + _format(recommended.value)
    )


def _format_total_diff(item: ResourceScan, resource: ResourceType, pods_current: int) -> str:
    selector = "requests"
    allocated = getattr(item.object.allocations, selector)[resource]
    recommended = getattr(item.recommended, selector)[resource]

    return __calc_diff(allocated, recommended, selector, pods_current)


@formatters.register()
def csv_export(result: Result) -> str:
    
    current_datetime = datetime.datetime.now().strftime("%Y%m%d%H%M%S")
    file_path = f"krr-{current_datetime}.csv"

    # We need to order the resource columns so that they are in the format of Namespace,Name,Pods,Old Pods,Type,Container,CPU Diff,CPU Requests,CPU Limits,Memory Diff,Memory Requests,Memory Limits
    resource_columns = []
    for resource in ResourceType:
        resource_columns.append(f"{resource.name} Diff")
        resource_columns.append(f"{resource.name} Requests")
        resource_columns.append(f"{resource.name} Limits")

    with open(file_path, 'w+', newline='') as csvfile:
        csv_writer = csv.writer(csvfile)
        csv_writer.writerow([
            "Namespace", "Name", "Pods", "Old Pods", "Type", "Container",
            *resource_columns

        ])

        for _, group in itertools.groupby(
            enumerate(result.scans), key=lambda x: (x[1].object.cluster, x[1].object.namespace, x[1].object.name)
        ):
            group_items = list(group)

            for j, (i, item) in enumerate(group_items):
                full_info_row = j == 0

                row = [
                    item.object.namespace if full_info_row else "",
                    item.object.name if full_info_row else "",
                    f"{item.object.current_pods_count}" if full_info_row else "",
                    f"{item.object.deleted_pods_count}" if full_info_row else "",
                    item.object.kind if full_info_row else "",
                    item.object.container,
                ]

                for resource in ResourceType:
                    row.append(_format_total_diff(item, resource, item.object.current_pods_count))
                    row += [_format_request_str(item, resource, selector) for selector in ["requests", "limits"]]

                csv_writer.writerow(row)
             
    logger.info("CSV File: %s", file_path)
    return ""