Skip to content
This repository has been archived by the owner on Aug 9, 2024. It is now read-only.

Add support for using local charms in integration tests #18

Merged
merged 2 commits into from
Aug 24, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 6 additions & 6 deletions src/utils/confeditor.py
Original file line number Diff line number Diff line change
Expand Up @@ -525,7 +525,7 @@ def auth_alt_types(self) -> Optional[List[str]]:
@auth_alt_types.setter
def auth_alt_types(self, value: Union[str, List[str]]) -> None:
"""Set configuration value for parameter `AuthAltTypes`."""
value = [value] if type(value) == str else value
value = [value] if isinstance(value, str) else value
self._metadata[_SlurmdbdToken.AuthAltTypes] = ",".join(value)

@auth_alt_types.deleter
Expand Down Expand Up @@ -600,7 +600,7 @@ def communication_parameters(self) -> Optional[List[str]]:
@communication_parameters.setter
def communication_parameters(self, value: Union[str, List[str]]) -> None:
"""Set configuration value for parameter `CommunicationParameters`."""
value = [value] if type(value) == str else value
value = [value] if isinstance(value, str) else value
self._metadata[_SlurmdbdToken.CommunicationParameters] = ",".join(value)

@communication_parameters.deleter
Expand Down Expand Up @@ -685,7 +685,7 @@ def debug_flags(self) -> Optional[List[str]]:
@debug_flags.setter
def debug_flags(self, value: Union[str, List[str]]) -> None:
"""Set configuration value for parameter `DebugFlags`."""
value = [value] if type(value) == str else value
value = [value] if isinstance(value, str) else value
for flag in value:
_check_debug_flag(flag)
self._metadata[_SlurmdbdToken.DebugFlags] = ",".join(value)
Expand Down Expand Up @@ -820,7 +820,7 @@ def parameters(self) -> Optional[List[str]]:
@parameters.setter
def parameters(self, value: Union[str, List[str]]) -> None:
"""Set configuration value for parameter `Parameters`."""
value = [value] if type(value) == str else value
value = [value] if isinstance(value, str) else value
self._metadata[_SlurmdbdToken.Parameters] = ",".join(value)

@parameters.deleter
Expand Down Expand Up @@ -855,7 +855,7 @@ def plugin_dir(self) -> Optional[List[str]]:
@plugin_dir.setter
def plugin_dir(self, value: Union[str, List[str]]) -> None:
"""Set configuration value for parameter `PluginDir`."""
value = [value] if type(value) == str else value
value = [value] if isinstance(value, str) else value
self._metadata[_SlurmdbdToken.PluginDir] = ":".join(value)

@plugin_dir.deleter
Expand All @@ -875,7 +875,7 @@ def private_data(self) -> Optional[List[str]]:
@private_data.setter
def private_data(self, value: Union[str, List[str]]) -> None:
"""Set configuration value for parameter `PrivateData`."""
value = [value] if type(value) == str else value
value = [value] if isinstance(value, str) else value
for data in value:
_check_private_data(data)
self._metadata[_SlurmdbdToken.PrivateData] = ",".join(value)
Expand Down
55 changes: 41 additions & 14 deletions tests/integration/conftest.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,35 +13,62 @@
# See the License for the specific language governing permissions and
# limitations under the License.

"""Configure integration test run."""
"""Configure slurmdbd operator integration tests."""

import pathlib
import logging
import os
from pathlib import Path
from typing import Union

import pytest
from _pytest.config.argparsing import Parser
from helpers import ETCD
from pytest_operator.plugin import OpsTest

logger = logging.getLogger(__name__)
SLURMCTLD_DIR = Path(os.getenv("SLURMCTLD_DIR", "../slurmctld-operator"))

def pytest_addoption(parser: Parser) -> None:

def pytest_addoption(parser) -> None:
parser.addoption(
"--charm-base",
action="store",
default="ubuntu@22.04",
help="Charm base version to use for integration tests",
)
parser.addoption(
"--charm-base", action="store", default="ubuntu@22.04", help="Charm base to test."
"--use-local",
action="store_true",
default=False,
help="Use SLURM operators located on localhost rather than pull from Charmhub",
)


@pytest.fixture(scope="module")
def charm_base(request) -> str:
"""Get slurmdbd charm base to use."""
return request.config.getoption("--charm-base")
return request.config.option.charm_base


@pytest.fixture(scope="module")
async def slurmdbd_charm(ops_test: OpsTest) -> Path:
"""Pack slurmdbd charm to use for integration tests."""
return await ops_test.build_charm(".")


@pytest.fixture(scope="module")
async def slurmdbd_charm(ops_test: OpsTest):
"""Build slurmdbd charm to use for integration tests."""
charm = await ops_test.build_charm(".")
return charm
async def slurmctld_charm(request, ops_test: OpsTest) -> Union[str, Path]:
"""Pack slurmctld charm to use for integration tests when --use-local is specified.

Returns:
`str` "slurmctld" if --use-local not specified or if SLURMD_DIR does not exist.
"""
if request.config.option.use_local:
logger.info("Using local slurmctld operator rather than pulling from Charmhub")
if SLURMCTLD_DIR.exists():
return await ops_test.build_charm(SLURMCTLD_DIR)
else:
logger.warning(
f"{SLURMCTLD_DIR} not found. "
f"Defaulting to latest/edge slurmctld operator from Charmhub"
)

def pytest_sessionfinish(session, exitstatus) -> None:
"""Clean up repository after test session has completed."""
pathlib.Path(ETCD).unlink(missing_ok=True)
return "slurmctld"
34 changes: 0 additions & 34 deletions tests/integration/helpers.py

This file was deleted.

28 changes: 12 additions & 16 deletions tests/integration/test_charm.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,16 +13,13 @@
# See the License for the specific language governing permissions and
# limitations under the License.

"""Test slurmdbd charm against other SLURM charms in the latest/edge channel."""
"""Test slurmdbd charm against other SLURM operators."""

import asyncio
import logging
import pathlib
from typing import Any, Coroutine

import pytest
import tenacity
from helpers import get_slurmctld_res
from pytest_operator.plugin import OpsTest

logger = logging.getLogger(__name__)
Expand All @@ -31,30 +28,31 @@
SLURMCTLD = "slurmctld"
DATABASE = "mysql"
ROUTER = "mysql-router"
UNIT_NAME = f"{SLURMDBD}/0"


@pytest.mark.abort_on_fail
@pytest.mark.skip_if_deployed
@pytest.mark.order(1)
async def test_build_and_deploy_against_edge(
ops_test: OpsTest, slurmdbd_charm: Coroutine[Any, Any, pathlib.Path], charm_base: str
ops_test: OpsTest, charm_base: str, slurmdbd_charm, slurmctld_charm
) -> None:
"""Test that the slurmdbd charm can stabilize against slurmctld and MySQL."""
logger.info(f"Deploying {SLURMDBD} against {SLURMCTLD} and {DATABASE}")
slurmctld_res = get_slurmctld_res()
# Pack charms
slurmdbd, slurmctld = await asyncio.gather(slurmdbd_charm, slurmctld_charm)
await asyncio.gather(
ops_test.model.deploy(
str(await slurmdbd_charm),
str(slurmdbd),
application_name=SLURMDBD,
num_units=1,
base=charm_base,
),
ops_test.model.deploy(
SLURMCTLD,
str(slurmctld),
application_name=SLURMCTLD,
channel="edge",
channel="edge" if isinstance(slurmctld, str) else None,
num_units=1,
resources=slurmctld_res,
base=charm_base,
),
ops_test.model.deploy(
Expand All @@ -72,16 +70,14 @@ async def test_build_and_deploy_against_edge(
base="ubuntu@22.04",
),
)
# Attach resources to charms.
await ops_test.juju("attach-resource", SLURMCTLD, f"etcd={slurmctld_res['etcd']}")
# Set relations for charmed applications.
await ops_test.model.integrate(f"{SLURMDBD}:{SLURMDBD}", f"{SLURMCTLD}:{SLURMDBD}")
await ops_test.model.integrate(f"{SLURMDBD}-{ROUTER}:backend-database", f"{DATABASE}:database")
await ops_test.model.integrate(f"{SLURMDBD}:database", f"{SLURMDBD}-{ROUTER}:database")
# Reduce the update status frequency to accelerate the triggering of deferred events.
async with ops_test.fast_forward():
await ops_test.model.wait_for_idle(apps=[SLURMDBD], status="active", timeout=1000)
assert ops_test.model.applications[SLURMDBD].units[0].workload_status == "active"
assert ops_test.model.units.get(UNIT_NAME).workload_status == "active"


@pytest.mark.abort_on_fail
Expand All @@ -94,7 +90,7 @@ async def test_build_and_deploy_against_edge(
async def test_slurmdbd_is_active(ops_test: OpsTest) -> None:
"""Test that slurmdbd is active inside Juju unit."""
logger.info("Checking that slurmdbd daemon is active inside unit")
slurmdbd_unit = ops_test.model.applications[SLURMDBD].units[0]
slurmdbd_unit = ops_test.model.units.get(UNIT_NAME)
res = (await slurmdbd_unit.ssh("systemctl is-active slurmdbd")).strip("\n")
assert res == "active"

Expand All @@ -109,7 +105,7 @@ async def test_slurmdbd_is_active(ops_test: OpsTest) -> None:
async def test_slurmdbd_port_listen(ops_test: OpsTest) -> None:
"""Test that slurmdbd is listening on port 6819."""
logger.info("Checking that slurmdbd is listening on port 6819")
slurmdbd_unit = ops_test.model.applications[SLURMDBD].units[0]
slurmdbd_unit = ops_test.model.units.get(UNIT_NAME)
res = await slurmdbd_unit.ssh("sudo lsof -t -n -iTCP:6819 -sTCP:LISTEN")
assert res != ""

Expand All @@ -124,6 +120,6 @@ async def test_slurmdbd_port_listen(ops_test: OpsTest) -> None:
async def test_munge_is_active(ops_test: OpsTest) -> None:
"""Test that munge is active inside Juju unit."""
logger.info("Checking that munge is active inside Juju unit")
slurmdbd_unit = ops_test.model.applications[SLURMDBD].units[0]
slurmdbd_unit = ops_test.model.units.get(UNIT_NAME)
res = (await slurmdbd_unit.ssh("systemctl is-active munge")).strip("\n")
assert res == "active"
3 changes: 2 additions & 1 deletion tox.ini
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ passenv =
PYTHONPATH
CHARM_BUILD_DIR
MODEL_SETTINGS
SLURMCTLD_DIR

[testenv:fmt]
description = Apply coding style standards to code
Expand Down Expand Up @@ -56,7 +57,7 @@ commands =
[testenv:integration]
description = Run integration tests
deps =
juju==3.1.0.1
juju
pytest==7.2.0
pytest-operator==0.26.0
pytest-order==1.1.0
Expand Down