2022-07-05 07:18:37 +00:00
|
|
|
import logging
|
|
|
|
import os
|
2023-11-27 10:03:18 +00:00
|
|
|
import random
|
2022-07-05 07:18:37 +00:00
|
|
|
import shutil
|
2023-10-23 14:11:53 +00:00
|
|
|
from datetime import datetime, timedelta, timezone
|
2023-11-22 16:58:49 +00:00
|
|
|
from importlib.metadata import entry_points
|
2023-05-15 09:59:33 +00:00
|
|
|
from typing import Optional
|
2022-07-05 07:18:37 +00:00
|
|
|
|
2022-09-20 15:03:52 +00:00
|
|
|
import allure
|
2022-07-05 07:18:37 +00:00
|
|
|
import pytest
|
2022-10-09 20:01:59 +00:00
|
|
|
import yaml
|
2023-10-23 14:11:53 +00:00
|
|
|
from dateutil import parser
|
2023-11-29 13:34:59 +00:00
|
|
|
from frostfs_testlib import plugins, reporter
|
2024-03-11 16:34:54 +00:00
|
|
|
from frostfs_testlib.credentials.interfaces import CredentialsProvider, User
|
2023-10-31 11:28:44 +00:00
|
|
|
from frostfs_testlib.healthcheck.interfaces import Healthcheck
|
2023-02-27 16:54:27 +00:00
|
|
|
from frostfs_testlib.hosting import Hosting
|
2023-11-29 13:34:59 +00:00
|
|
|
from frostfs_testlib.reporter import AllureHandler, StepsLogger
|
2023-05-15 09:59:33 +00:00
|
|
|
from frostfs_testlib.resources.common import (
|
2022-11-28 19:27:37 +00:00
|
|
|
ASSETS_DIR,
|
2022-12-07 12:38:56 +00:00
|
|
|
COMPLEX_OBJECT_CHUNKS_COUNT,
|
|
|
|
COMPLEX_OBJECT_TAIL_SIZE,
|
|
|
|
SIMPLE_OBJECT_SIZE,
|
2022-12-06 19:57:29 +00:00
|
|
|
)
|
2023-10-31 14:51:09 +00:00
|
|
|
from frostfs_testlib.s3 import AwsCliClient, Boto3ClientWrapper, S3ClientWrapper, VersioningStatus
|
2023-05-15 09:59:33 +00:00
|
|
|
from frostfs_testlib.shell import LocalShell, Shell
|
2024-05-23 07:33:20 +00:00
|
|
|
from frostfs_testlib.steps.cli.container import DEFAULT_EC_PLACEMENT_RULE, DEFAULT_PLACEMENT_RULE
|
2023-05-15 09:59:33 +00:00
|
|
|
from frostfs_testlib.steps.cli.object import get_netmap_netinfo
|
|
|
|
from frostfs_testlib.steps.s3 import s3_helper
|
2023-11-22 16:58:49 +00:00
|
|
|
from frostfs_testlib.storage import get_service_registry
|
2023-09-29 13:16:06 +00:00
|
|
|
from frostfs_testlib.storage.cluster import Cluster, ClusterNode
|
2023-05-26 04:49:23 +00:00
|
|
|
from frostfs_testlib.storage.controllers.cluster_state_controller import ClusterStateController
|
2023-10-23 14:11:53 +00:00
|
|
|
from frostfs_testlib.storage.dataclasses.frostfs_services import StorageNode
|
2023-08-02 11:54:03 +00:00
|
|
|
from frostfs_testlib.storage.dataclasses.object_size import ObjectSize
|
2024-03-11 12:49:37 +00:00
|
|
|
from frostfs_testlib.storage.dataclasses.policy import PlacementPolicy
|
2024-03-11 16:34:54 +00:00
|
|
|
from frostfs_testlib.storage.dataclasses.wallet import WalletInfo
|
2023-05-15 09:59:33 +00:00
|
|
|
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
2023-09-29 13:16:06 +00:00
|
|
|
from frostfs_testlib.testing.parallel import parallel
|
2023-10-23 14:11:53 +00:00
|
|
|
from frostfs_testlib.testing.test_control import wait_for_success
|
2023-05-15 09:59:33 +00:00
|
|
|
from frostfs_testlib.utils import env_utils, version_utils
|
2024-06-06 18:03:38 +00:00
|
|
|
from frostfs_testlib.utils.file_utils import generate_file
|
2023-05-15 09:59:33 +00:00
|
|
|
|
|
|
|
from pytest_tests.resources.common import HOSTING_CONFIG_FILE, TEST_CYCLES_COUNT
|
2022-09-05 09:35:46 +00:00
|
|
|
|
|
|
|
logger = logging.getLogger("NeoLogger")
|
2022-07-05 07:18:37 +00:00
|
|
|
|
2024-01-10 10:43:52 +00:00
|
|
|
SERVICE_ACTIVE_TIME = 20
|
|
|
|
|
2023-03-14 09:21:40 +00:00
|
|
|
# Add logs check test even if it's not fit to mark selectors
|
|
|
|
def pytest_configure(config: pytest.Config):
|
|
|
|
markers = config.option.markexpr
|
2024-03-18 09:16:10 +00:00
|
|
|
if markers != "" and "sanity" not in markers:
|
2023-03-14 09:21:40 +00:00
|
|
|
config.option.markexpr = f"logs_after_session or ({markers})"
|
|
|
|
|
|
|
|
|
2023-11-29 13:34:59 +00:00
|
|
|
number_key = pytest.StashKey[str]()
|
|
|
|
start_time = pytest.StashKey[int]()
|
|
|
|
test_outcome = pytest.StashKey[str]()
|
2023-03-14 09:21:40 +00:00
|
|
|
# pytest hook. Do not rename
|
2023-03-17 08:49:58 +00:00
|
|
|
def pytest_collection_modifyitems(items: list[pytest.Item]):
|
2024-06-24 14:51:36 +00:00
|
|
|
# Change order of tests based on @pytest.mark.order(<int>) marker
|
|
|
|
def order(item: pytest.Item) -> int:
|
|
|
|
order_marker = item.get_closest_marker("order")
|
|
|
|
if order_marker and (len(order_marker.args) != 1 or not isinstance(order_marker.args[0], int)):
|
|
|
|
raise RuntimeError("Incorrect usage of pytest.mark.order")
|
2022-11-03 13:17:08 +00:00
|
|
|
|
2024-06-24 14:51:36 +00:00
|
|
|
order_value = order_marker.args[0] if order_marker else 0
|
|
|
|
return order_value
|
|
|
|
|
|
|
|
items.sort(key=lambda item: order(item))
|
2022-07-05 07:18:37 +00:00
|
|
|
|
2022-11-07 07:10:11 +00:00
|
|
|
|
2023-11-29 13:34:59 +00:00
|
|
|
# pytest hook. Do not rename
|
|
|
|
def pytest_collection_finish(session: pytest.Session):
|
|
|
|
items_total = len(session.items)
|
|
|
|
for number, item in enumerate(session.items, 1):
|
|
|
|
item.stash[number_key] = f"[{number}/{items_total}]"
|
|
|
|
item.stash[test_outcome] = ""
|
2023-12-05 08:33:59 +00:00
|
|
|
item.stash[start_time] = 0
|
2023-11-29 13:34:59 +00:00
|
|
|
|
|
|
|
|
|
|
|
# pytest hook. Do not rename
|
|
|
|
def pytest_runtest_setup(item: pytest.Item):
|
|
|
|
item.stash[start_time] = int(datetime.now().timestamp())
|
|
|
|
logger.info(f"STARTED {item.stash[number_key]}: {item.name}")
|
|
|
|
|
|
|
|
|
|
|
|
# pytest hook. Do not rename
|
|
|
|
def pytest_runtest_makereport(item: pytest.Item, call: pytest.CallInfo):
|
|
|
|
if call.excinfo is not None:
|
2023-12-05 08:33:59 +00:00
|
|
|
if call.excinfo.typename == "Skipped":
|
|
|
|
item.stash[start_time] = int(datetime.now().timestamp())
|
|
|
|
item.stash[test_outcome] += f"SKIPPED on {call.when}; "
|
|
|
|
else:
|
|
|
|
item.stash[test_outcome] += f"FAILED on {call.when}; "
|
2023-11-29 13:34:59 +00:00
|
|
|
|
|
|
|
if call.when == "teardown":
|
|
|
|
duration = int(datetime.now().timestamp()) - item.stash[start_time]
|
|
|
|
if not item.stash[test_outcome]:
|
|
|
|
outcome = "PASSED "
|
|
|
|
else:
|
|
|
|
outcome = item.stash[test_outcome]
|
|
|
|
logger.info(f"ENDED {item.stash[number_key]}: {item.name}: {outcome}(duration={duration}s)")
|
|
|
|
|
|
|
|
|
|
|
|
# pytest hook. Do not rename
|
2023-03-17 08:49:58 +00:00
|
|
|
def pytest_generate_tests(metafunc: pytest.Metafunc):
|
|
|
|
if (
|
|
|
|
TEST_CYCLES_COUNT <= 1
|
|
|
|
or metafunc.definition.get_closest_marker("logs_after_session")
|
|
|
|
or metafunc.definition.get_closest_marker("no_cycles")
|
|
|
|
):
|
|
|
|
return
|
|
|
|
|
|
|
|
metafunc.fixturenames.append("cycle")
|
|
|
|
metafunc.parametrize(
|
|
|
|
"cycle",
|
|
|
|
range(1, TEST_CYCLES_COUNT + 1),
|
|
|
|
ids=[f"cycle {cycle}" for cycle in range(1, TEST_CYCLES_COUNT + 1)],
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2022-10-13 18:53:44 +00:00
|
|
|
@pytest.fixture(scope="session")
|
2022-10-13 16:13:45 +00:00
|
|
|
def configure_testlib():
|
2023-11-29 13:34:59 +00:00
|
|
|
reporter.get_reporter().register_handler(AllureHandler())
|
|
|
|
reporter.get_reporter().register_handler(StepsLogger())
|
|
|
|
|
2023-05-15 09:59:33 +00:00
|
|
|
logging.getLogger("paramiko").setLevel(logging.INFO)
|
2023-11-22 16:58:49 +00:00
|
|
|
|
|
|
|
# Register Services for cluster
|
|
|
|
registry = get_service_registry()
|
|
|
|
services = entry_points(group="frostfs.testlib.services")
|
|
|
|
for svc in services:
|
|
|
|
registry.register_service(svc.name, svc.load())
|
|
|
|
|
2022-10-13 16:13:45 +00:00
|
|
|
yield
|
2022-10-13 18:53:44 +00:00
|
|
|
|
|
|
|
|
2022-09-05 09:35:46 +00:00
|
|
|
@pytest.fixture(scope="session")
|
2022-10-13 16:13:45 +00:00
|
|
|
def client_shell(configure_testlib) -> Shell:
|
|
|
|
yield LocalShell()
|
2022-08-01 06:16:36 +00:00
|
|
|
|
|
|
|
|
2022-10-09 20:01:59 +00:00
|
|
|
@pytest.fixture(scope="session")
|
2022-10-13 16:13:45 +00:00
|
|
|
def hosting(configure_testlib) -> Hosting:
|
2022-10-09 20:01:59 +00:00
|
|
|
with open(HOSTING_CONFIG_FILE, "r") as file:
|
|
|
|
hosting_config = yaml.full_load(file)
|
2022-07-26 07:28:13 +00:00
|
|
|
|
2022-10-09 20:01:59 +00:00
|
|
|
hosting_instance = Hosting()
|
|
|
|
hosting_instance.configure(hosting_config)
|
2022-12-05 22:31:45 +00:00
|
|
|
|
2022-10-09 20:01:59 +00:00
|
|
|
yield hosting_instance
|
2022-07-05 07:18:37 +00:00
|
|
|
|
|
|
|
|
2022-10-13 16:13:45 +00:00
|
|
|
@pytest.fixture(scope="session")
|
|
|
|
def require_multiple_hosts(hosting: Hosting):
|
2022-10-14 16:56:15 +00:00
|
|
|
"""Designates tests that require environment with multiple hosts.
|
|
|
|
|
|
|
|
These tests will be skipped on an environment that has only 1 host.
|
|
|
|
"""
|
2022-10-13 16:13:45 +00:00
|
|
|
if len(hosting.hosts) <= 1:
|
|
|
|
pytest.skip("Test only works with multiple hosts")
|
|
|
|
yield
|
|
|
|
|
|
|
|
|
2023-10-25 12:56:59 +00:00
|
|
|
@pytest.fixture(scope="session")
|
|
|
|
def require_multiple_interfaces(cluster: Cluster):
|
|
|
|
"""
|
|
|
|
We determine that there are the required number of interfaces for tests
|
|
|
|
|
|
|
|
If there are no required interfaces, the tests will be skipped.
|
|
|
|
"""
|
|
|
|
interfaces = cluster.cluster_nodes[0].host.config.interfaces
|
|
|
|
if "internal1" not in interfaces or "data1" not in interfaces:
|
|
|
|
pytest.skip("This test requires multiple internal and data interfaces")
|
|
|
|
yield
|
|
|
|
|
|
|
|
|
2022-12-07 12:38:56 +00:00
|
|
|
@pytest.fixture(scope="session")
|
|
|
|
def max_object_size(cluster: Cluster, client_shell: Shell) -> int:
|
|
|
|
storage_node = cluster.storage_nodes[0]
|
2024-03-11 16:34:54 +00:00
|
|
|
wallet = WalletInfo.from_node(storage_node)
|
2022-12-07 12:38:56 +00:00
|
|
|
net_info = get_netmap_netinfo(
|
2024-03-11 16:34:54 +00:00
|
|
|
wallet=wallet,
|
2022-12-07 12:38:56 +00:00
|
|
|
endpoint=storage_node.get_rpc_endpoint(),
|
|
|
|
shell=client_shell,
|
|
|
|
)
|
|
|
|
yield net_info["maximum_object_size"]
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(scope="session")
|
2023-09-08 10:35:34 +00:00
|
|
|
def simple_object_size(max_object_size: int) -> ObjectSize:
|
2023-08-02 11:54:03 +00:00
|
|
|
size = min(int(SIMPLE_OBJECT_SIZE), max_object_size)
|
|
|
|
return ObjectSize("simple", size)
|
2022-12-07 12:38:56 +00:00
|
|
|
|
|
|
|
|
2024-06-06 18:03:38 +00:00
|
|
|
@pytest.fixture()
|
|
|
|
def file_path(object_size: ObjectSize) -> str:
|
2024-06-24 23:27:54 +00:00
|
|
|
return generate_file(object_size.value)
|
2024-06-06 18:03:38 +00:00
|
|
|
|
|
|
|
|
2022-12-07 12:38:56 +00:00
|
|
|
@pytest.fixture(scope="session")
|
2023-09-08 10:35:34 +00:00
|
|
|
def complex_object_size(max_object_size: int) -> ObjectSize:
|
2023-08-02 11:54:03 +00:00
|
|
|
size = max_object_size * int(COMPLEX_OBJECT_CHUNKS_COUNT) + int(COMPLEX_OBJECT_TAIL_SIZE)
|
|
|
|
return ObjectSize("complex", size)
|
2022-12-07 12:38:56 +00:00
|
|
|
|
|
|
|
|
2023-09-08 10:35:34 +00:00
|
|
|
# By default we want all tests to be executed with both object sizes
|
|
|
|
# This can be overriden in choosen tests if needed
|
2023-10-31 13:47:32 +00:00
|
|
|
@pytest.fixture(
|
|
|
|
scope="session",
|
|
|
|
params=[pytest.param("simple", marks=pytest.mark.simple), pytest.param("complex", marks=pytest.mark.complex)],
|
|
|
|
)
|
2023-09-08 10:35:34 +00:00
|
|
|
def object_size(
|
|
|
|
simple_object_size: ObjectSize, complex_object_size: ObjectSize, request: pytest.FixtureRequest
|
|
|
|
) -> ObjectSize:
|
|
|
|
if request.param == "simple":
|
|
|
|
return simple_object_size
|
|
|
|
|
|
|
|
return complex_object_size
|
|
|
|
|
2024-05-23 07:33:20 +00:00
|
|
|
|
2024-03-11 12:49:37 +00:00
|
|
|
@pytest.fixture(scope="session")
|
|
|
|
def rep_placement_policy() -> PlacementPolicy:
|
|
|
|
return PlacementPolicy("rep", DEFAULT_PLACEMENT_RULE)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(scope="session")
|
|
|
|
def ec_placement_policy() -> PlacementPolicy:
|
|
|
|
return PlacementPolicy("ec", DEFAULT_EC_PLACEMENT_RULE)
|
|
|
|
|
|
|
|
|
|
|
|
# By default we want all tests to be executed with both storage policies.
|
|
|
|
# This can be overriden in choosen tests if needed.
|
|
|
|
@pytest.fixture(
|
|
|
|
scope="session",
|
|
|
|
params=[pytest.param("rep", marks=pytest.mark.rep), pytest.param("ec", marks=pytest.mark.ec)],
|
|
|
|
)
|
|
|
|
def placement_policy(
|
|
|
|
rep_placement_policy: PlacementPolicy, ec_placement_policy: PlacementPolicy, request: pytest.FixtureRequest
|
|
|
|
) -> PlacementPolicy:
|
|
|
|
if request.param == "rep":
|
|
|
|
return rep_placement_policy
|
|
|
|
|
|
|
|
return ec_placement_policy
|
2023-09-08 10:35:34 +00:00
|
|
|
|
2024-05-23 07:33:20 +00:00
|
|
|
|
2022-12-05 22:31:45 +00:00
|
|
|
@pytest.fixture(scope="session")
|
2023-05-15 09:59:33 +00:00
|
|
|
def cluster(temp_directory: str, hosting: Hosting, client_shell: Shell) -> Cluster:
|
2022-12-23 08:52:05 +00:00
|
|
|
cluster = Cluster(hosting)
|
2023-05-15 09:59:33 +00:00
|
|
|
if cluster.is_local_devenv():
|
2022-12-23 08:52:05 +00:00
|
|
|
cluster.create_wallet_configs(hosting)
|
2023-05-15 09:59:33 +00:00
|
|
|
|
|
|
|
ClusterTestBase.shell = client_shell
|
|
|
|
ClusterTestBase.cluster = cluster
|
|
|
|
|
2022-12-23 08:52:05 +00:00
|
|
|
yield cluster
|
2022-11-10 14:56:25 +00:00
|
|
|
|
|
|
|
|
2024-06-24 23:27:54 +00:00
|
|
|
@allure.title("[Session]: Provide S3 policy")
|
|
|
|
@pytest.fixture(scope="session")
|
2023-05-15 09:59:33 +00:00
|
|
|
def s3_policy(request: pytest.FixtureRequest):
|
|
|
|
policy = None
|
|
|
|
if "param" in request.__dict__:
|
|
|
|
policy = request.param
|
|
|
|
|
|
|
|
return policy
|
|
|
|
|
|
|
|
|
2023-05-26 04:49:23 +00:00
|
|
|
@pytest.fixture(scope="session")
|
2023-10-31 11:28:44 +00:00
|
|
|
@allure.title("[Session] Create healthcheck object")
|
|
|
|
def healthcheck(cluster: Cluster) -> Healthcheck:
|
2023-11-22 16:58:49 +00:00
|
|
|
healthcheck_cls = plugins.load_plugin(
|
2023-10-31 11:28:44 +00:00
|
|
|
"frostfs.testlib.healthcheck", cluster.cluster_nodes[0].host.config.healthcheck_plugin_name
|
|
|
|
)
|
|
|
|
|
|
|
|
return healthcheck_cls()
|
|
|
|
|
|
|
|
|
2024-06-05 10:07:07 +00:00
|
|
|
@pytest.fixture(scope="session")
|
|
|
|
def cluster_state_controller_session(
|
|
|
|
client_shell: Shell, cluster: Cluster, healthcheck: Healthcheck
|
|
|
|
) -> ClusterStateController:
|
2023-10-31 11:28:44 +00:00
|
|
|
controller = ClusterStateController(client_shell, cluster, healthcheck)
|
2024-06-05 10:07:07 +00:00
|
|
|
return controller
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def cluster_state_controller(cluster_state_controller_session: ClusterStateController) -> ClusterStateController:
|
|
|
|
yield cluster_state_controller_session
|
|
|
|
cluster_state_controller_session.start_stopped_hosts()
|
|
|
|
cluster_state_controller_session.start_all_stopped_services()
|
2023-05-26 04:49:23 +00:00
|
|
|
|
|
|
|
|
2024-03-11 16:34:54 +00:00
|
|
|
@pytest.fixture(scope="session")
|
|
|
|
def credentials_provider(cluster: Cluster) -> CredentialsProvider:
|
|
|
|
return CredentialsProvider(cluster)
|
|
|
|
|
|
|
|
|
2024-06-24 23:27:54 +00:00
|
|
|
@allure.title("[Session]: Create S3 client")
|
2023-10-31 14:51:09 +00:00
|
|
|
@pytest.fixture(
|
2024-06-24 23:27:54 +00:00
|
|
|
scope="session",
|
2023-10-31 14:51:09 +00:00
|
|
|
params=[
|
2024-06-24 23:27:54 +00:00
|
|
|
pytest.param(AwsCliClient, marks=[pytest.mark.aws, pytest.mark.weekly]),
|
|
|
|
pytest.param(Boto3ClientWrapper, marks=[pytest.mark.boto3, pytest.mark.nightly]),
|
2023-10-31 14:51:09 +00:00
|
|
|
],
|
|
|
|
)
|
2023-05-15 09:59:33 +00:00
|
|
|
def s3_client(
|
2024-03-11 16:34:54 +00:00
|
|
|
default_user: User,
|
2023-05-15 09:59:33 +00:00
|
|
|
s3_policy: Optional[str],
|
|
|
|
cluster: Cluster,
|
|
|
|
request: pytest.FixtureRequest,
|
2024-03-11 16:34:54 +00:00
|
|
|
credentials_provider: CredentialsProvider,
|
2023-05-15 09:59:33 +00:00
|
|
|
) -> S3ClientWrapper:
|
2024-02-29 23:26:56 +00:00
|
|
|
node = cluster.cluster_nodes[0]
|
2024-03-11 16:34:54 +00:00
|
|
|
credentials_provider.S3.provide(default_user, node, s3_policy)
|
2023-05-15 09:59:33 +00:00
|
|
|
|
|
|
|
s3_client_cls = request.param
|
2024-03-11 16:34:54 +00:00
|
|
|
client = s3_client_cls(
|
|
|
|
default_user.s3_credentials.access_key, default_user.s3_credentials.secret_key, cluster.default_s3_gate_endpoint
|
|
|
|
)
|
2024-02-29 23:26:56 +00:00
|
|
|
return client
|
2023-05-15 09:59:33 +00:00
|
|
|
|
|
|
|
|
2023-08-29 13:43:16 +00:00
|
|
|
@pytest.fixture
|
|
|
|
def versioning_status(request: pytest.FixtureRequest) -> VersioningStatus:
|
|
|
|
if "param" in request.__dict__:
|
|
|
|
return request.param
|
|
|
|
|
|
|
|
return VersioningStatus.UNDEFINED
|
|
|
|
|
|
|
|
|
2024-06-24 23:27:54 +00:00
|
|
|
def unique_name(prefix: str) -> str:
|
|
|
|
return f"{prefix}{hex(int(datetime.now().timestamp() * 1000000))}"
|
|
|
|
|
|
|
|
|
|
|
|
@allure.title("[Session] Bulk create buckets for tests")
|
|
|
|
@pytest.fixture(scope="session")
|
|
|
|
def buckets_pool(s3_client: S3ClientWrapper, request: pytest.FixtureRequest):
|
|
|
|
test_buckets: list = []
|
|
|
|
|
|
|
|
s3_client_type = type(s3_client).__name__
|
|
|
|
|
|
|
|
for test in request.session.items:
|
|
|
|
if s3_client_type not in test.name:
|
|
|
|
continue
|
|
|
|
|
|
|
|
if "bucket" in test.fixturenames:
|
|
|
|
test_buckets.append(unique_name("bucket-"))
|
|
|
|
|
|
|
|
if "two_buckets" in test.fixturenames:
|
|
|
|
test_buckets.append(unique_name("bucket-"))
|
|
|
|
test_buckets.append(unique_name("bucket-"))
|
2023-11-01 16:20:15 +00:00
|
|
|
|
2024-06-24 23:27:54 +00:00
|
|
|
if test_buckets:
|
|
|
|
parallel(s3_client.create_bucket, test_buckets)
|
|
|
|
|
|
|
|
return test_buckets
|
|
|
|
|
|
|
|
|
|
|
|
@allure.title("[Test] Create bucket")
|
|
|
|
@pytest.fixture
|
|
|
|
def bucket(buckets_pool: list[str], s3_client: S3ClientWrapper, versioning_status: VersioningStatus):
|
|
|
|
if buckets_pool:
|
|
|
|
bucket_name = buckets_pool.pop()
|
|
|
|
else:
|
|
|
|
bucket_name = s3_client.create_bucket()
|
2023-05-15 09:59:33 +00:00
|
|
|
|
|
|
|
if versioning_status:
|
|
|
|
s3_helper.set_bucket_versioning(s3_client, bucket_name, versioning_status)
|
|
|
|
|
2024-06-24 23:27:54 +00:00
|
|
|
return bucket_name
|
2023-05-15 09:59:33 +00:00
|
|
|
|
|
|
|
|
2024-06-24 23:27:54 +00:00
|
|
|
@allure.title("[Test] Create two buckets")
|
2023-05-15 09:59:33 +00:00
|
|
|
@pytest.fixture
|
2024-06-24 23:27:54 +00:00
|
|
|
def two_buckets(buckets_pool: list[str], s3_client: S3ClientWrapper) -> list[str]:
|
|
|
|
buckets: list[str] = []
|
2023-10-31 14:51:09 +00:00
|
|
|
|
2024-06-24 23:27:54 +00:00
|
|
|
for _ in range(2):
|
|
|
|
if buckets_pool:
|
|
|
|
buckets.append(buckets_pool.pop())
|
|
|
|
else:
|
|
|
|
buckets.append(s3_client.create_bucket())
|
|
|
|
|
|
|
|
return buckets
|
2023-05-15 09:59:33 +00:00
|
|
|
|
|
|
|
|
2024-06-07 14:13:50 +00:00
|
|
|
@allure.title("[Autouse/Session] Collect binary versions")
|
2022-10-09 20:01:59 +00:00
|
|
|
@pytest.fixture(scope="session", autouse=True)
|
2024-06-07 14:13:50 +00:00
|
|
|
def collect_binary_versions(hosting: Hosting, client_shell: Shell, request: pytest.FixtureRequest):
|
2023-05-15 09:59:33 +00:00
|
|
|
local_versions = version_utils.get_local_binaries_versions(client_shell)
|
2024-06-07 14:13:50 +00:00
|
|
|
remote_versions = version_utils.get_remote_binaries_versions(hosting)
|
|
|
|
remote_versions_keys = list(remote_versions.keys())
|
2022-07-05 07:18:37 +00:00
|
|
|
|
2023-11-29 13:34:59 +00:00
|
|
|
all_versions = {
|
|
|
|
**local_versions,
|
2024-06-07 14:13:50 +00:00
|
|
|
**{
|
|
|
|
f"{name}_{remote_versions_keys.index(host) + 1:02d}": version
|
|
|
|
for host, versions in remote_versions.items()
|
|
|
|
for name, version in versions.items()
|
|
|
|
},
|
2023-11-29 13:34:59 +00:00
|
|
|
}
|
2023-05-15 09:59:33 +00:00
|
|
|
|
|
|
|
environment_dir = request.config.getoption("--alluredir")
|
|
|
|
if not environment_dir:
|
|
|
|
return None
|
|
|
|
|
|
|
|
file_path = f"{environment_dir}/environment.properties"
|
|
|
|
env_utils.save_env_properties(file_path, all_versions)
|
2022-08-03 15:20:50 +00:00
|
|
|
|
|
|
|
|
2023-11-29 13:34:59 +00:00
|
|
|
@reporter.step("Prepare tmp directory")
|
2022-09-05 09:35:46 +00:00
|
|
|
@pytest.fixture(scope="session")
|
2023-11-29 13:34:59 +00:00
|
|
|
def temp_directory(configure_testlib):
|
|
|
|
with reporter.step("Prepare tmp directory"):
|
2022-11-10 14:56:25 +00:00
|
|
|
full_path = os.path.join(os.getcwd(), ASSETS_DIR)
|
|
|
|
shutil.rmtree(full_path, ignore_errors=True)
|
|
|
|
os.mkdir(full_path)
|
|
|
|
|
2022-08-29 20:42:34 +00:00
|
|
|
yield full_path
|
2022-11-10 14:56:25 +00:00
|
|
|
|
2023-11-29 13:34:59 +00:00
|
|
|
with reporter.step("Remove tmp directory"):
|
2022-11-10 14:56:25 +00:00
|
|
|
shutil.rmtree(full_path)
|
2022-08-29 20:42:34 +00:00
|
|
|
|
|
|
|
|
2023-11-29 13:34:59 +00:00
|
|
|
@reporter.step("[Autouse/Session] Test session start time")
|
2022-08-29 20:42:34 +00:00
|
|
|
@pytest.fixture(scope="session", autouse=True)
|
2023-11-29 13:34:59 +00:00
|
|
|
def session_start_time(configure_testlib):
|
2022-08-26 19:58:56 +00:00
|
|
|
start_time = datetime.utcnow()
|
2023-03-14 09:21:40 +00:00
|
|
|
return start_time
|
2022-11-15 08:39:07 +00:00
|
|
|
|
2022-08-26 19:58:56 +00:00
|
|
|
|
2023-10-23 14:11:53 +00:00
|
|
|
@allure.title("[Autouse/Session] After deploy healthcheck")
|
|
|
|
@pytest.fixture(scope="session", autouse=True)
|
|
|
|
def after_deploy_healthcheck(cluster: Cluster):
|
2023-11-29 13:34:59 +00:00
|
|
|
with reporter.step("Wait for cluster readiness after deploy"):
|
2023-10-23 14:11:53 +00:00
|
|
|
parallel(readiness_on_node, cluster.cluster_nodes)
|
|
|
|
|
|
|
|
|
2023-11-29 13:34:59 +00:00
|
|
|
@wait_for_success(60 * SERVICE_ACTIVE_TIME * 3, 60, title="Wait for {cluster_node} readiness")
|
2023-10-23 14:11:53 +00:00
|
|
|
def readiness_on_node(cluster_node: ClusterNode):
|
2024-01-12 15:16:56 +00:00
|
|
|
if (
|
|
|
|
"skip_readiness_check" in cluster_node.host.config.attributes
|
|
|
|
and cluster_node.host.config.attributes["skip_readiness_check"]
|
|
|
|
):
|
|
|
|
return
|
|
|
|
|
2023-10-23 14:11:53 +00:00
|
|
|
# TODO: Move to healtcheck classes
|
|
|
|
svc_name = cluster_node.service(StorageNode).get_service_systemctl_name()
|
2023-11-29 13:34:59 +00:00
|
|
|
with reporter.step(f"Check service {svc_name} is active"):
|
2023-10-23 14:11:53 +00:00
|
|
|
result = cluster_node.host.get_shell().exec(f"systemctl is-active {svc_name}")
|
|
|
|
assert "active" == result.stdout.strip(), f"Service {svc_name} should be in active state"
|
|
|
|
|
2023-11-29 13:34:59 +00:00
|
|
|
with reporter.step(f"Check service {svc_name} is active more than {SERVICE_ACTIVE_TIME} minutes"):
|
2023-10-23 14:11:53 +00:00
|
|
|
result = cluster_node.host.get_shell().exec(
|
|
|
|
f"systemctl show {svc_name} --property ActiveEnterTimestamp | cut -d '=' -f 2"
|
|
|
|
)
|
|
|
|
start_time = parser.parse(result.stdout.strip())
|
|
|
|
current_time = datetime.now(tz=timezone.utc)
|
|
|
|
active_time = current_time - start_time
|
|
|
|
|
|
|
|
active_minutes = active_time.seconds // 60
|
|
|
|
active_seconds = active_time.seconds - active_minutes * 60
|
|
|
|
|
|
|
|
assert active_time > timedelta(
|
|
|
|
minutes=SERVICE_ACTIVE_TIME
|
|
|
|
), f"Service should be in active state more than {SERVICE_ACTIVE_TIME} minutes, current {active_minutes}m:{active_seconds}s"
|
|
|
|
|
|
|
|
|
2024-03-11 16:34:54 +00:00
|
|
|
@reporter.step("Prepare default user with wallet")
|
|
|
|
@pytest.fixture(scope="session")
|
|
|
|
def default_user(credentials_provider: CredentialsProvider, cluster: Cluster) -> User:
|
|
|
|
# always unique username
|
|
|
|
user = User(f"user_{hex(int(datetime.now().timestamp() * 1000000))}")
|
|
|
|
node = cluster.cluster_nodes[0]
|
|
|
|
|
|
|
|
credentials_provider.GRPC.provide(user, node)
|
|
|
|
|
|
|
|
return user
|
|
|
|
|
|
|
|
|
|
|
|
@reporter.step("Get wallet for default user")
|
2022-11-28 19:27:37 +00:00
|
|
|
@pytest.fixture(scope="session")
|
2024-03-11 16:34:54 +00:00
|
|
|
def default_wallet(default_user: User) -> WalletInfo:
|
|
|
|
return default_user.wallet
|
2023-09-18 13:56:53 +00:00
|
|
|
|
|
|
|
|
2023-11-27 10:03:18 +00:00
|
|
|
@pytest.fixture()
|
|
|
|
@allure.title("Select random node for testing")
|
|
|
|
def node_under_test(cluster: Cluster) -> ClusterNode:
|
|
|
|
selected_node = random.choice(cluster.cluster_nodes)
|
2023-11-29 13:34:59 +00:00
|
|
|
reporter.attach(f"{selected_node}", "Selected node")
|
2023-11-27 10:03:18 +00:00
|
|
|
return selected_node
|