From 3b1f3f1666e2a7ce2205370e992bd2adcf479620 Mon Sep 17 00:00:00 2001 From: Kamil Gierszewski Date: Fri, 13 Dec 2024 05:28:37 +0100 Subject: [PATCH] tests: Refactor ioclass stat tests Signed-off-by: Kamil Gierszewski --- .../tests/stats/test_ioclass_stats.py | 654 ++++++++++++------ 1 file changed, 433 insertions(+), 221 deletions(-) diff --git a/test/functional/tests/stats/test_ioclass_stats.py b/test/functional/tests/stats/test_ioclass_stats.py index 5f0396366..fcdd851ac 100644 --- a/test/functional/tests/stats/test_ioclass_stats.py +++ b/test/functional/tests/stats/test_ioclass_stats.py @@ -1,335 +1,547 @@ # # Copyright(c) 2019-2021 Intel Corporation +# Copyright(c) 2024 Huawei Technologies Co., Ltd. # SPDX-License-Identifier: BSD-3-Clause # -import random +import random import pytest from api.cas import casadm from api.cas import ioclass_config -from api.cas.cache_config import CleaningPolicy, CacheMode, CacheLineSize +from api.cas.cache_config import ( + CleaningPolicy, + CacheMode, + CacheLineSize, + CacheModeTrait, + SeqCutOffPolicy, +) from api.cas.casadm import StatsFilter from api.cas.cli_messages import ( check_stderr_msg, get_stats_ioclass_id_not_configured, - get_stats_ioclass_id_out_of_range + get_stats_ioclass_id_out_of_range, ) +from api.cas.ioclass_config import IoClass from api.cas.statistics import ( - config_stats_ioclass, - usage_stats, - usage_stats_ioclass, - request_stats, - block_stats_core, - block_stats_cache + UsageStats, + RequestStatsChunk, + BlockStats, + RequestStats, + IoClassUsageStats, + IoClassConfigStats, ) + from core.test_run import TestRun from storage_devices.disk import DiskType, DiskTypeSet, DiskTypeLowerThan from test_tools.disk_utils import Filesystem +from test_tools.fs_utils import create_directory from test_utils.filesystem.file import File from test_utils.os_utils import sync, Udev from test_utils.output import CmdException from test_utils.size import Size, Unit -IoClass = ioclass_config.IoClass - -mountpoint = "/tmp/cas1-1" -cache_id = 1 - @pytest.mark.require_disk("cache", DiskTypeSet([DiskType.optane, DiskType.nand])) @pytest.mark.require_disk("core", DiskTypeLowerThan("cache")) +@pytest.mark.parametrizex( + "cache_mode", CacheMode.with_traits(CacheModeTrait.InsertWrite | CacheModeTrait.InsertRead) +) @pytest.mark.parametrize("random_cls", [random.choice(list(CacheLineSize))]) -def test_ioclass_stats_basic(random_cls): +def test_ioclass_stats_basic(cache_mode: CacheMode, random_cls: CacheLineSize): """ - title: Basic test for retrieving IO class statistics. - description: | - Check if statistics are retrieved only for configured IO classes. - pass_criteria: - - Statistics are retrieved for configured IO classes. - - Error is displayed when retrieving statistics for non-configured IO class. - - Error is displayed when retrieving statistics for out of range IO class id. + title: Basic test for retrieving IO class statistics. + description: | + Check if statistics are retrieved only for configured IO classes. + pass_criteria: + - Statistics are retrieved for configured IO classes. + - Error is displayed when retrieving statistics for non-configured IO class. + - Error is displayed when retrieving statistics for out of range IO class id. """ - min_ioclass_id = 11 max_ioclass_id = 21 + mountpoint = "/tmp/open_cas_test_data/cas1-1" + file_size_base = Unit.Blocks4096.value + IoClass = ioclass_config.IoClass - with TestRun.step("Test prepare"): - prepare(random_cls) + with TestRun.step("Prepare cache and core devices"): + cache_device = TestRun.disks["cache"] + core_device = TestRun.disks["core"] - with TestRun.step("Prepare IO class config file"): - ioclass_list = [] - for class_id in range(min_ioclass_id, max_ioclass_id): - ioclass_list.append(IoClass( + cache_device.create_partitions([Size(500, Unit.MebiByte)]) + core_device.create_partitions([Size(2, Unit.GibiByte)]) + + cache_device = cache_device.partitions[0] + core_device = core_device.partitions[0] + + with TestRun.step("Create filesystem on core device"): + core_device.create_filesystem(Filesystem.ext4) + + with TestRun.step("Disable udev"): + Udev.disable() + + with TestRun.step("Start cache"): + cache = casadm.start_cache( + cache_dev=cache_device, + cache_mode=cache_mode, + cache_line_size=random_cls, + force=True, + ) + cache_id = cache.cache_id + + with TestRun.step("Set cleaning policy to NOP and seqcutoff to never on cache"): + cache.set_cleaning_policy(CleaningPolicy.nop) + cache.set_seq_cutoff_policy(SeqCutOffPolicy.never) + + with TestRun.step(f"Add core to cache"): + core = cache.add_core(core_dev=core_device) + + with TestRun.step("Reset counter for each core"): + core.reset_counters() + + with TestRun.step("Create mountpoint directory"): + create_directory(path=mountpoint) + + with TestRun.step("Prepare IO class config file and save it"): + ioclass_list = [ + IoClass( class_id=class_id, rule=f"file_size:le:{4096 * class_id}&done", - priority=22 - )) + priority=22, + ) + for class_id in range(min_ioclass_id, max_ioclass_id) + ] + IoClass.save_list_to_config_file(ioclass_list, True) + + with TestRun.step("Load IO class config file"): + cache.load_io_class(ioclass_config.default_config_file_path) + + with TestRun.step("Generate files with particular sizes in temporary folder"): + file_list = [] + for class_id in range(min_ioclass_id, max_ioclass_id): + path = f"/tmp/open_cas_test_data/test_file_{file_size_base * class_id}" + File.create_file(path) + file = File(path) + file.padding(Size(file_size_base * class_id, Unit.Byte)) + file_list.append(file) + + with TestRun.step("Mount core"): + core.mount(mountpoint) + + with TestRun.step("Prepare IO class config file and save it"): + ioclass_list = [ + IoClass(class_id=class_id, rule=f"file_size:le:{4096 * class_id}&done", priority=22) + for class_id in range(min_ioclass_id, max_ioclass_id) + ] IoClass.save_list_to_config_file(ioclass_list, True) with TestRun.step("Load IO class config file"): casadm.load_io_classes(cache_id, file=ioclass_config.default_config_file_path) - with TestRun.step("Try retrieving IO class stats for all allowed id values " - "and one out of range id"): + with TestRun.step( + "Try retrieving IO class stats for allowed ids values and one out of range id" + ): for class_id in range(ioclass_config.MAX_IO_CLASS_ID + 2): out_of_range = " out of range" if class_id > ioclass_config.MAX_IO_CLASS_ID else "" - with TestRun.group(f"Checking{out_of_range} IO class id {class_id}..."): + + with TestRun.group(f"Checking {out_of_range} IO class id {class_id}..."): expected = class_id == 0 or class_id in range(min_ioclass_id, max_ioclass_id) + try: - casadm.print_statistics( - cache_id=cache_id, - io_class_id=class_id, - per_io_class=True) + casadm.print_statistics(cache_id=cache_id, io_class_id=class_id, io_class=True) if not expected: TestRun.LOGGER.error( - f"Stats retrieved for not configured IO class {class_id}") + f"Stats retrieved for not configured IO class {class_id}" + ) except CmdException as e: if expected: TestRun.LOGGER.error(f"Stats not retrieved for IO class id: {class_id}") elif class_id <= ioclass_config.MAX_IO_CLASS_ID: if not check_stderr_msg(e.output, get_stats_ioclass_id_not_configured): TestRun.LOGGER.error( - f"Wrong message for unused IO class id: {class_id}") + f"Wrong message for unused IO class id: {class_id}" + ) elif not check_stderr_msg(e.output, get_stats_ioclass_id_out_of_range): TestRun.LOGGER.error( - f"Wrong message for out of range IO class id: {class_id}") + f"Wrong message for out of range IO class id: {class_id}" + ) @pytest.mark.require_disk("cache", DiskTypeSet([DiskType.optane, DiskType.nand])) @pytest.mark.require_disk("core", DiskTypeLowerThan("cache")) +@pytest.mark.parametrizex( + "cache_mode", CacheMode.with_traits(CacheModeTrait.InsertWrite | CacheModeTrait.InsertRead) +) @pytest.mark.parametrize("random_cls", [random.choice(list(CacheLineSize))]) -def test_ioclass_stats_sum(random_cls): +def test_ioclass_stats_sum(cache_mode: CacheMode, random_cls: CacheLineSize): """ - title: Test for sum of IO class statistics. - description: | - Check if statistics for configured IO classes sum up to cache/core statistics. - pass_criteria: - - Per class cache IO class statistics sum up to cache statistics. - - Per class core IO class statistics sum up to core statistics. + title: Test for sum of IO class statistics. + description: | + Check if statistics for configured IO classes sum up to cache/core statistics. + pass_criteria: + - Per class cache IO class statistics sum up to cache statistics. + - Per class core IO class statistics sum up to core statistics. """ - - min_ioclass_id = 1 - max_ioclass_id = 11 + min_ioclass_id = 11 + max_ioclass_id = 22 + mountpoint = "/tmp/open_cas_test_data/cas1-1" file_size_base = Unit.Blocks4096.value + IoClass = ioclass_config.IoClass - with TestRun.step("Test prepare"): - caches, cores = prepare(random_cls) - cache, core = caches[0], cores[0] + with TestRun.step("Prepare cache and core devices"): + cache_device = TestRun.disks["cache"] + core_device = TestRun.disks["core"] - with TestRun.step("Prepare IO class config file"): - ioclass_list = [] - for class_id in range(min_ioclass_id, max_ioclass_id): - ioclass_list.append(IoClass( + cache_device.create_partitions([Size(500, Unit.MebiByte)]) + core_device.create_partitions([Size(2, Unit.GibiByte)]) + + cache_device = cache_device.partitions[0] + core_device = core_device.partitions[0] + + with TestRun.step("Create filesystem on core device"): + core_device.create_filesystem(Filesystem.ext4) + + with TestRun.step("Disable udev"): + Udev.disable() + + with TestRun.step("Start cache"): + cache = casadm.start_cache( + cache_dev=cache_device, + cache_mode=cache_mode, + cache_line_size=random_cls, + force=True, + ) + + with TestRun.step("Set cleaning policy to NOP and seqcutoff to never on cache"): + cache.set_cleaning_policy(CleaningPolicy.nop) + cache.set_seq_cutoff_policy(SeqCutOffPolicy.never) + + with TestRun.step(f"Add core to cache"): + core = cache.add_core(core_dev=core_device) + + with TestRun.step("Reset counter for each core"): + core.reset_counters() + + with TestRun.step("Create mountpoint directory"): + create_directory(path=mountpoint) + + with TestRun.step("Prepare IO class config file and save it"): + ioclass_list = [ + IoClass( class_id=class_id, rule=f"file_size:le:{file_size_base * class_id}&done", - priority=22 - )) + priority=22, + ) + for class_id in range(min_ioclass_id, max_ioclass_id) + ] IoClass.save_list_to_config_file(ioclass_list, True) with TestRun.step("Load IO class config file"): cache.load_io_class(ioclass_config.default_config_file_path) with TestRun.step("Generate files with particular sizes in temporary folder"): - files_list = [] + file_list = [] for class_id in range(min_ioclass_id, max_ioclass_id): - path = f"/tmp/test_file_{file_size_base * class_id}" + path = f"/tmp/open_cas_test_data/test_file_{file_size_base * class_id}" File.create_file(path) - f = File(path) - f.padding(Size(file_size_base * class_id, Unit.Byte)) - files_list.append(f) + file = File(path) + file.padding(Size(file_size_base * class_id, Unit.Byte)) + file_list.append(file) - with TestRun.step("Copy files to mounted core"): + with TestRun.step("Mount core"): core.mount(mountpoint) - for f in files_list: - TestRun.LOGGER.info(f"Copying file {f.name} to mounted core") - f.copy(mountpoint) + + with TestRun.step("Copy files to mounted core"): + for file in file_list: + file.copy(mountpoint) sync() + # To prevent stats pollution by filesystem requests, umount core device # after files are copied core.unmount() sync() with TestRun.step("Check if per class cache IO class statistics sum up to cache statistics"): - # Name of stats, which should not be compared - not_compare_stats = ["clean", "occupancy", "free"] ioclass_id_list = list(range(min_ioclass_id, max_ioclass_id)) # Append default IO class id ioclass_id_list.append(0) - cache_stats = cache.get_statistics_flat( + cache_stats = cache.get_statistics( stat_filter=[StatsFilter.usage, StatsFilter.req, StatsFilter.blk] ) - for ioclass_id in ioclass_id_list: - ioclass_stats = cache.get_statistics_flat( - stat_filter=[StatsFilter.usage, StatsFilter.req, StatsFilter.blk], + ioclass_stats_list = [ + cache.get_io_class_statistics( io_class_id=ioclass_id, + stat_filter=[StatsFilter.usage, StatsFilter.req, StatsFilter.blk], ) - for stat_name in cache_stats: - if stat_name in not_compare_stats: - continue - cache_stats[stat_name] -= ioclass_stats[stat_name] - - for stat_name in cache_stats: - if stat_name in not_compare_stats: - continue - stat_val = ( - cache_stats[stat_name].get_value() - if isinstance(cache_stats[stat_name], Size) - else cache_stats[stat_name] - ) - if stat_val != 0: - TestRun.LOGGER.error(f"{stat_name} diverged for cache!\n") + for ioclass_id in ioclass_id_list + ] + + for ioclass_stats in ioclass_stats_list: + for cache_stat, io_class_stat in zip(cache_stats, ioclass_stats): + # Name of stats, which should not be compared + # UsageStats: clean, occupancy, free + if isinstance(cache_stat, UsageStats): + cache_stat.dirty -= io_class_stat.dirty + else: + cache_stat -= io_class_stat + + stat_val = [] + for cache_stat in cache_stats: + if isinstance(cache_stat, UsageStats): + stat_val.append(cache_stat.dirty.get_value()) + stat_name = cache_stat + else: + for stat_name in cache_stat: + if isinstance(stat_name, Size): + stat_val.append(stat_name.get_value()) + elif isinstance(stat_name, RequestStatsChunk): + stat_val.append([stat for stat in stat_name]) + else: + stat_val.append(stat_name) + + if all(stat_val) != 0: + TestRun.LOGGER.error(f"Diverged for cache!\n {stat_name}") with TestRun.step("Check if per class core IO class statistics sum up to core statistics"): - core_stats = core.get_statistics_flat( + core_stats = core.get_statistics( stat_filter=[StatsFilter.usage, StatsFilter.req, StatsFilter.blk] ) - for ioclass_id in ioclass_id_list: - ioclass_stats = core.get_statistics_flat( - stat_filter=[StatsFilter.usage, StatsFilter.req, StatsFilter.blk], + ioclass_stats_list = [ + core.get_io_class_statistics( io_class_id=ioclass_id, + stat_filter=[StatsFilter.usage, StatsFilter.req, StatsFilter.blk], ) - for stat_name in core_stats: - if stat_name in not_compare_stats: - continue - core_stats[stat_name] -= ioclass_stats[stat_name] - - for stat_name in core_stats: - if stat_name in not_compare_stats: - continue - stat_val = ( - core_stats[stat_name].get_value() - if isinstance(core_stats[stat_name], Size) - else core_stats[stat_name] - ) - if stat_val != 0: - TestRun.LOGGER.error(f"{stat_name} diverged for core!\n") - - with TestRun.step("Test cleanup"): - for f in files_list: - f.remove() + for ioclass_id in ioclass_id_list + ] + + for ioclass_stats in ioclass_stats_list: + for core_stat, io_class_stat in zip(core_stats, ioclass_stats): + # Name of stats, which should not be compared + # UsageStats: clean, occupancy, free + if isinstance(core_stat, UsageStats): + core_stat.dirty -= io_class_stat.dirty + else: + core_stat -= io_class_stat + + stat_val = [] + for core_stat in core_stats: + if isinstance(core_stat, UsageStats): + stat_val.append(core_stat.dirty.get_value()) + stat_name = core_stat + else: + for stat_name in core_stat: + if isinstance(stat_name, Size): + stat_val.append(stat_name.get_value()) + elif isinstance(stat_name, RequestStatsChunk): + stat_val.append([stat for stat in stat_name]) + else: + stat_val.append(stat_name) + + if all(stat_val) != 0: + TestRun.LOGGER.error(f"Diverged for cache!\n {stat_name}") @pytest.mark.require_disk("cache", DiskTypeSet([DiskType.optane, DiskType.nand])) @pytest.mark.require_disk("core", DiskTypeLowerThan("cache")) -@pytest.mark.parametrize("stat_filter", [StatsFilter.req, StatsFilter.usage, StatsFilter.conf, - StatsFilter.blk]) -@pytest.mark.parametrize("per_core", [True, False]) @pytest.mark.parametrize("random_cls", [random.choice(list(CacheLineSize))]) -def test_ioclass_stats_sections(stat_filter, per_core, random_cls): +def test_ioclass_stats_sections(random_cls): """ - title: Test for cache/core IO class statistics sections. - description: | - Check if IO class statistics sections for cache/core print all required entries and - no additional ones. - pass_criteria: - - Section statistics contain all required entries. - - Section statistics do not contain any additional entries. + title: Test for cache/core IO class statistics sections. + description: | + Check if IO class statistics sections for cache/core print all required entries and + no additional ones. + pass_criteria: + - Section statistics contain all required entries. """ - with TestRun.step("Test prepare"): - caches, cores = prepare(random_cls, cache_count=4, cores_per_cache=3) - - with TestRun.step(f"Validate displayed {stat_filter.name} statistics for default IO class for " - f"{'cores' if per_core else 'caches'}"): - for cache in caches: - with TestRun.group(f"Cache {cache.cache_id}"): - for core in cache.get_core_devices(): - if per_core: - TestRun.LOGGER.info(f"Core {core.cache_id}-{core.core_id}") - statistics = ( - core.get_statistics_flat( - io_class_id=0, stat_filter=[stat_filter]) if per_core - else cache.get_statistics_flat( - io_class_id=0, stat_filter=[stat_filter])) - validate_statistics(statistics, stat_filter, per_core) - if not per_core: - break + cache_count = 4 + cores_per_cache = 3 + ioclass_stat_filter = [StatsFilter.req, StatsFilter.usage, StatsFilter.conf, StatsFilter.blk] + cache_modes = [CacheMode.WT, CacheMode.WB, CacheMode.WA, CacheMode.WO] + mountpoint = "/tmp/open_cas_test_data/cas1-1" - with TestRun.step("Load random IO class configuration for each cache"): - for cache in caches: - random_list = IoClass.generate_random_ioclass_list(ioclass_config.MAX_IO_CLASS_ID + 1) - IoClass.save_list_to_config_file(random_list, add_default_rule=False) - cache.load_io_class(ioclass_config.default_config_file_path) + with TestRun.step("Prepare cache and core devices"): + cache_device = TestRun.disks["cache"] + core_device = TestRun.disks["core"] - with TestRun.step(f"Validate displayed {stat_filter.name} statistics for every configured IO " - f"class for all {'cores' if per_core else 'caches'}"): - for cache in caches: - with TestRun.group(f"Cache {cache.cache_id}"): - for core in cache.get_core_devices(): - core_info = f"Core {core.cache_id}-{core.core_id} ," if per_core else "" - for class_id in range(ioclass_config.MAX_IO_CLASS_ID + 1): - with TestRun.group(core_info + f"IO class id {class_id}"): - statistics = ( - core.get_statistics_flat(class_id, [stat_filter]) if per_core - else cache.get_statistics_flat(class_id, [stat_filter])) - validate_statistics(statistics, stat_filter, per_core) - if stat_filter == StatsFilter.conf: # no percentage statistics for conf - continue - statistics_percents = ( - core.get_statistics_flat( - class_id, [stat_filter], percentage_val=True) if per_core - else cache.get_statistics_flat( - class_id, [stat_filter], percentage_val=True)) - validate_statistics(statistics_percents, stat_filter, per_core) - if not per_core: - break - - -def get_checked_statistics(stat_filter: StatsFilter, per_core: bool): - if stat_filter == StatsFilter.conf: - return config_stats_ioclass - if stat_filter == StatsFilter.usage: - return usage_stats_ioclass - if stat_filter == StatsFilter.blk: - return block_stats_core if per_core else block_stats_cache - if stat_filter == StatsFilter.req: - return request_stats - - -def validate_statistics(statistics: dict, stat_filter: StatsFilter, per_core: bool): - for stat_name in get_checked_statistics(stat_filter, per_core): - if stat_name not in statistics.keys(): - TestRun.LOGGER.error(f"Value for {stat_name} not displayed in output") - else: - del statistics[stat_name] - if len(statistics.keys()): - TestRun.LOGGER.error(f"Additional statistics found: {', '.join(statistics.keys())}") - - -def prepare(random_cls, cache_count=1, cores_per_cache=1): - cache_modes = [CacheMode.WT, CacheMode.WB, CacheMode.WA, CacheMode.WO] - ioclass_config.remove_ioclass_config() + cache_device.create_partitions([Size(500, Unit.MebiByte)] * cache_count) + core_device.create_partitions([Size(2, Unit.GibiByte)] * cache_count * cores_per_cache) - cache_device = TestRun.disks['cache'] - core_device = TestRun.disks['core'] + cache_devices = cache_device.partitions + core_devices = core_device.partitions - cache_device.create_partitions([Size(500, Unit.MebiByte)] * cache_count) - core_device.create_partitions([Size(2, Unit.GibiByte)] * cache_count * cores_per_cache) + with TestRun.step("Create filesystem on each core device"): + for core_device in core_devices: + core_device.create_filesystem(Filesystem.ext4) - cache_devices = cache_device.partitions - core_devices = core_device.partitions - for core_device in core_devices: - core_device.create_filesystem(Filesystem.ext4) + with TestRun.step("Disable udev"): + Udev.disable() - Udev.disable() - caches, cores = [], [] - for i, cache_device in enumerate(cache_devices): - TestRun.LOGGER.info(f"Starting cache on {cache_device.path}") - cache = casadm.start_cache(cache_device, - force=True, - cache_mode=cache_modes[i], - cache_line_size=random_cls) - caches.append(cache) - TestRun.LOGGER.info("Setting cleaning policy to NOP") - cache.set_cleaning_policy(CleaningPolicy.nop) - for core_device in core_devices[i * cores_per_cache:(i + 1) * cores_per_cache]: - TestRun.LOGGER.info( - f"Adding core device {core_device.path} to cache {cache.cache_id}") - core = cache.add_core(core_dev=core_device) + with TestRun.step("Start cache for each cache device"): + cache_list = [ + casadm.start_cache( + cache_dev=device, + cache_mode=cache_mode, + cache_line_size=random_cls, + force=True, + ) + for device, cache_mode in zip(cache_devices, cache_modes) + ] + + with TestRun.step("Set cleaning policy to NOP and seqcutoff to never on cache"): + for cache in cache_list: + cache.set_cleaning_policy(CleaningPolicy.nop) + cache.set_seq_cutoff_policy(SeqCutOffPolicy.never) + + with TestRun.step(f"Add {cores_per_cache} core to each cache"): + core_list = [cache.add_core(core_dev=core_dev) for core_dev in core_devices] + + with TestRun.step("Reset counter for each core"): + for core in core_list: core.reset_counters() - cores.append(core) - TestRun.executor.run_expect_success(f"mkdir -p {mountpoint}") + with TestRun.step("Create mountpoint directory"): + create_directory(path=mountpoint) + + with TestRun.step(f"Validate displayed statistics for default IO class on each cache"): + for cache in cache_list: + statistics = cache.get_io_class_statistics( + io_class_id=0, stat_filter=ioclass_stat_filter + ) + for stat_name in ioclass_stat_filter: + io_class_stat_name = get_checked_ioclass_statistics(stat_name) + if not any(isinstance(statistic, io_class_stat_name) for statistic in statistics): + TestRun.LOGGER.error(f"Value for {stat_name} not displayed in output") + + if not len(ioclass_stat_filter) != len([statistics]): + parsed_filter = ", ".join([(str(stat) + "_stats") for stat in ioclass_stat_filter]) + parsed_stats = ", ".join(vars(statistics)) + TestRun.LOGGER.error( + f"Wrong number of statistics in output\n" + f"Expected:{parsed_filter}\n" + f"Got: {parsed_stats}" + ) + + with TestRun.step(f"Validate displayed statistics for default IO class on each core"): + for cache in cache_list: + cache_core_device_list = [core for core in core_list if core.cache_id == cache.cache_id] + for core in cache_core_device_list: + statistics = core.get_io_class_statistics( + io_class_id=0, stat_filter=ioclass_stat_filter + ) + + for stat_name in ioclass_stat_filter: + io_class_stat_name = get_checked_ioclass_statistics(stat_name) + if not any( + isinstance(statistic, io_class_stat_name) for statistic in statistics + ): + TestRun.LOGGER.error(f"Value for {stat_name} not displayed in output") + + if not len(ioclass_stat_filter) != len([statistics]): + parsed_filter = ", ".join( + [(str(stat) + "_stats") for stat in ioclass_stat_filter] + ) + parsed_stats = ", ".join(vars(statistics)) + TestRun.LOGGER.error( + f"Wrong number of statistics in output\n" + f"Expected:{parsed_filter}\n" + f"Got: {parsed_stats}" + ) + + with TestRun.step("Load random IO class configuration for each cache"): + for cache in cache_list: + random_list = IoClass.generate_random_ioclass_list(ioclass_config.MAX_IO_CLASS_ID + 1) + IoClass.save_list_to_config_file(random_list, add_default_rule=False) + cache.load_io_class(ioclass_config.default_config_file_path) + + with TestRun.step(f"Validate displayed statistics for configured IO class on each cache"): + for cache in cache_list: + statistics = cache.get_io_class_statistics( + io_class_id=0, stat_filter=ioclass_stat_filter + ) + for stat_name in ioclass_stat_filter: + io_class_stat_name = get_checked_ioclass_statistics(stat_name) + if not any(isinstance(statistic, io_class_stat_name) for statistic in statistics): + TestRun.LOGGER.error(f"Value for {stat_name} not displayed in output") - return caches, cores + statistics = cache.get_io_class_statistics( + io_class_id=0, stat_filter=ioclass_stat_filter, percentage_val=True + ) + for stat_name in ioclass_stat_filter: + io_class_stat_name = get_checked_ioclass_statistics(stat_name) + if not any(isinstance(statistic, io_class_stat_name) for statistic in statistics): + TestRun.LOGGER.error(f"Percentage for {stat_name} not displayed in output") + + if not len(ioclass_stat_filter) != len([statistics]): + parsed_filter = ", ".join([(str(stat) + "_stats") for stat in ioclass_stat_filter]) + parsed_stats = ", ".join(vars(statistics)) + TestRun.LOGGER.error( + f"Wrong number of statistics in output\n" + f"Expected:{parsed_filter}\n" + f"Got: {parsed_stats}" + ) + + with TestRun.step(f"Validate displayed statistics for configured IO class on each core"): + for cache in cache_list: + cache_core_device_list = [core for core in core_list if core.cache_id == cache.cache_id] + for core in cache_core_device_list: + core_info = f"Core {core.cache_id}-{core.core_id}" + for class_id in range(ioclass_config.MAX_IO_CLASS_ID + 1): + statistics = cache.get_io_class_statistics( + io_class_id=class_id, stat_filter=ioclass_stat_filter + ) + for stat_name in ioclass_stat_filter: + io_class_stat_name = get_checked_ioclass_statistics(stat_name) + if not any( + isinstance(statistic, io_class_stat_name) for statistic in statistics + ): + TestRun.LOGGER.error( + f"Value for {core_info} - {stat_name} not displayed in output" + ) + + if not len(ioclass_stat_filter) != len([statistics]): + parsed_filter = ", ".join( + [(str(stat) + "_stats") for stat in ioclass_stat_filter] + ) + parsed_stats = ", ".join(vars(statistics)) + TestRun.LOGGER.error( + f"Wrong number of statistics in output\n" + f"Expected:{parsed_filter}\n" + f"Got: {parsed_stats}" + ) + + statistics = cache.get_io_class_statistics( + io_class_id=class_id, stat_filter=ioclass_stat_filter, percentage_val=True + ) + for stat_name in ioclass_stat_filter: + io_class_stat_name = get_checked_ioclass_statistics(stat_name) + if not any( + isinstance(statistic, io_class_stat_name) for statistic in statistics + ): + TestRun.LOGGER.error( + f"Percentage for {core_info} - {stat_name} not " + f"displayed in output" + ) + + if not len(ioclass_stat_filter) != len([statistics]): + parsed_filter = ", ".join( + [(str(stat) + "_stats") for stat in ioclass_stat_filter] + ) + parsed_stats = ", ".join(vars(statistics)) + TestRun.LOGGER.error( + f"Wrong number of statistics in output\n" + f"Expected:{parsed_filter}\n" + f"Got: {parsed_stats}" + ) + + +def get_checked_ioclass_statistics(stat_filter: StatsFilter): + match stat_filter: + case StatsFilter.conf: + return IoClassConfigStats + case StatsFilter.usage: + return IoClassUsageStats + case StatsFilter.req: + return RequestStats + case StatsFilter.blk: + return BlockStats