forked from TrueCloudLab/frostfs-testcases
98 lines
4.5 KiB
Python
98 lines
4.5 KiB
Python
|
import math
|
||
|
import re
|
||
|
|
||
|
import allure
|
||
|
import pytest
|
||
|
from frostfs_testlib import reporter
|
||
|
from frostfs_testlib.steps.cli.container import create_container, delete_container
|
||
|
from frostfs_testlib.steps.cli.object import delete_object, get_object_nodes, put_object_to_random_node
|
||
|
from frostfs_testlib.storage.cluster import Cluster, ClusterNode
|
||
|
from frostfs_testlib.storage.dataclasses.object_size import ObjectSize
|
||
|
from frostfs_testlib.storage.dataclasses.wallet import WalletInfo
|
||
|
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
||
|
from frostfs_testlib.testing.test_control import wait_for_success
|
||
|
from frostfs_testlib.utils.file_utils import generate_file
|
||
|
|
||
|
|
||
|
@pytest.mark.container
|
||
|
class TestContainerMetrics(ClusterTestBase):
|
||
|
@wait_for_success(interval=10)
|
||
|
def check_sum_counter_metrics_in_nodes(
|
||
|
self, cluster_nodes: list[ClusterNode], cid: str, phy_exp: int, logic_exp: int, user_exp: int
|
||
|
):
|
||
|
counter_phy = 0
|
||
|
counter_logic = 0
|
||
|
counter_user = 0
|
||
|
for cluster_node in cluster_nodes:
|
||
|
metric_result = cluster_node.metrics.storage.get_metric_container(f"container_objects_total", cid)
|
||
|
counter_phy += self.get_count_metric_type_from_stdout(metric_result.stdout, "phy")
|
||
|
counter_logic += self.get_count_metric_type_from_stdout(metric_result.stdout, "logic")
|
||
|
counter_user += self.get_count_metric_type_from_stdout(metric_result.stdout, "user")
|
||
|
|
||
|
assert counter_phy == phy_exp, f"Expected metric Phy={phy_exp}, Actual: {counter_phy} in nodes: {cluster_nodes}"
|
||
|
assert (
|
||
|
counter_logic == logic_exp
|
||
|
), f"Expected metric logic={logic_exp}, Actual: {counter_logic} in nodes: {cluster_nodes}"
|
||
|
assert (
|
||
|
counter_user == user_exp
|
||
|
), f"Expected metric User={user_exp}, Actual: {counter_user} in nodes: {cluster_nodes}"
|
||
|
|
||
|
@staticmethod
|
||
|
def get_count_metric_type_from_stdout(metric_result_stdout: str, metric_type: str):
|
||
|
result = re.findall(rf'type="{metric_type}"}}\s(\d+)', metric_result_stdout)
|
||
|
return sum(map(int, result))
|
||
|
|
||
|
@allure.title("Container metrics (obj_size={object_size})")
|
||
|
def test_container_metrics(
|
||
|
self, object_size: ObjectSize, max_object_size: int, default_wallet: WalletInfo, cluster: Cluster
|
||
|
):
|
||
|
file_path = generate_file(object_size.value)
|
||
|
placement_policy = "REP 2 IN X CBF 2 SELECT 2 FROM * AS X"
|
||
|
copies = 2
|
||
|
object_chunks = 0
|
||
|
head_object = 1
|
||
|
link_object = 0
|
||
|
if object_size.value > max_object_size:
|
||
|
object_chunks = math.ceil(object_size.value / max_object_size)
|
||
|
link_object = 1
|
||
|
|
||
|
with reporter.step(f"Create container with policy {placement_policy}"):
|
||
|
cid = create_container(
|
||
|
default_wallet,
|
||
|
rule=placement_policy,
|
||
|
shell=self.shell,
|
||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||
|
)
|
||
|
|
||
|
with reporter.step("Put object to random node"):
|
||
|
storage_object_id = put_object_to_random_node(
|
||
|
wallet=default_wallet,
|
||
|
path=file_path,
|
||
|
cid=cid,
|
||
|
shell=self.shell,
|
||
|
cluster=cluster,
|
||
|
)
|
||
|
|
||
|
with reporter.step("Check metric appears in node where the object is located"):
|
||
|
object_nodes = get_object_nodes(
|
||
|
cluster=cluster, cid=cid, oid=storage_object_id, alive_node=cluster.cluster_nodes[0]
|
||
|
)
|
||
|
count_metrics_exp = (object_chunks + head_object + link_object) * copies
|
||
|
self.check_sum_counter_metrics_in_nodes(
|
||
|
object_nodes, cid, phy_exp=count_metrics_exp, logic_exp=count_metrics_exp, user_exp=copies
|
||
|
)
|
||
|
|
||
|
with reporter.step("Delete file, wait until gc remove object"):
|
||
|
delete_object(default_wallet, cid, storage_object_id, self.shell, self.cluster.default_rpc_endpoint)
|
||
|
count_metrics_exp = len(object_nodes)
|
||
|
self.check_sum_counter_metrics_in_nodes(
|
||
|
object_nodes, cid, phy_exp=count_metrics_exp, logic_exp=count_metrics_exp, user_exp=0
|
||
|
)
|
||
|
|
||
|
with reporter.step("Check metrics(Phy, Logic, User) in each nodes"):
|
||
|
# Phy and Logic metrics are 4, because in rule 'CBF 2 SELECT 2 FROM', cbf2*sel2=4
|
||
|
self.check_sum_counter_metrics_in_nodes(cluster.cluster_nodes, cid, phy_exp=4, logic_exp=4, user_exp=0)
|
||
|
|
||
|
with reporter.step("Delete container"):
|
||
|
delete_container(default_wallet, cid, shell=self.shell, endpoint=self.cluster.default_rpc_endpoint)
|