mirror of
https://github.com/Cian-H/I-Form_Server_Node_Deployer.git
synced 2025-12-23 06:32:08 +00:00
Implemented basic unit tests
This commit is contained in:
@@ -11,6 +11,7 @@ CLIENT_STDOUT = true
|
|||||||
CLEANUP_IMAGES = true
|
CLEANUP_IMAGES = true
|
||||||
CLI = false
|
CLI = false
|
||||||
DEBUG = false
|
DEBUG = false
|
||||||
|
TESTING = false
|
||||||
|
|
||||||
[local]
|
[local]
|
||||||
FUELIGNITION_URL = "http://localhost:3000/fuel-ignition/edit"
|
FUELIGNITION_URL = "http://localhost:3000/fuel-ignition/edit"
|
||||||
@@ -25,6 +26,10 @@ CLI = true
|
|||||||
DEBUG = true
|
DEBUG = true
|
||||||
CLI = false
|
CLI = false
|
||||||
|
|
||||||
|
[test]
|
||||||
|
TESTING = true
|
||||||
|
CLEANUP_IMAGES = false
|
||||||
|
|
||||||
[default.snoop.install]
|
[default.snoop.install]
|
||||||
snoop = "ss"
|
snoop = "ss"
|
||||||
out = "snoop.log"
|
out = "snoop.log"
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ description = ""
|
|||||||
authors = ["Cian Hughes <cian.hughes@dcu.ie>"]
|
authors = ["Cian Hughes <cian.hughes@dcu.ie>"]
|
||||||
readme = "README.md"
|
readme = "README.md"
|
||||||
packages = [
|
packages = [
|
||||||
{include = "node_deployer", from = "src"}
|
{include = "node_deployer", from = "src"},
|
||||||
]
|
]
|
||||||
|
|
||||||
[tool.poetry.dependencies]
|
[tool.poetry.dependencies]
|
||||||
|
|||||||
@@ -0,0 +1,13 @@
|
|||||||
|
from . import (
|
||||||
|
config,
|
||||||
|
autoignition,
|
||||||
|
create_img,
|
||||||
|
create_disk,
|
||||||
|
)
|
||||||
|
|
||||||
|
__all__ = [
|
||||||
|
"config",
|
||||||
|
"autoignition",
|
||||||
|
"create_img",
|
||||||
|
"create_disk",
|
||||||
|
]
|
||||||
@@ -10,23 +10,26 @@ import git
|
|||||||
from selenium import webdriver
|
from selenium import webdriver
|
||||||
from selenium.webdriver.common.by import By
|
from selenium.webdriver.common.by import By
|
||||||
from selenium.webdriver.support import expected_conditions as EC
|
from selenium.webdriver.support import expected_conditions as EC
|
||||||
from selenium.webdriver.support.ui import WebDriverWait
|
from selenium.webdriver.support.ui import WebDriverWait # type: ignore
|
||||||
import typer
|
import typer
|
||||||
|
|
||||||
from .config import config
|
|
||||||
from .cli import cli_spinner
|
from .cli import cli_spinner
|
||||||
|
from .config import config
|
||||||
from .debug import debug_guard
|
from .debug import debug_guard
|
||||||
from .utils import ensure_build_dir
|
from .utils import ensure_build_dir, next_free_tcp_port
|
||||||
|
|
||||||
|
|
||||||
def create_driver() -> webdriver.Remote:
|
def create_driver(port: int) -> webdriver.Remote:
|
||||||
"""Creates a selenium webdriver instance
|
"""Creates a selenium webdriver instance
|
||||||
|
|
||||||
|
Args:
|
||||||
|
port (int): The port to connect to
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
webdriver.Remote: The created webdriver instance
|
webdriver.Remote: The created webdriver instance
|
||||||
"""
|
"""
|
||||||
driver = webdriver.Remote(
|
driver = webdriver.Remote(
|
||||||
"http://127.0.0.1:4444",
|
f"http://127.0.0.1:{port}",
|
||||||
options=webdriver.FirefoxOptions(),
|
options=webdriver.FirefoxOptions(),
|
||||||
)
|
)
|
||||||
driver.implicitly_wait(10)
|
driver.implicitly_wait(10)
|
||||||
@@ -34,7 +37,7 @@ def create_driver() -> webdriver.Remote:
|
|||||||
|
|
||||||
|
|
||||||
def convert_json_via_fuelignition(
|
def convert_json_via_fuelignition(
|
||||||
container: docker.models.containers.Container,
|
container: docker.models.containers.Container, # type: ignore
|
||||||
driver: webdriver.Remote,
|
driver: webdriver.Remote,
|
||||||
fuelignition_json: Path,
|
fuelignition_json: Path,
|
||||||
img_path: Path,
|
img_path: Path,
|
||||||
@@ -88,7 +91,7 @@ def convert_json_via_fuelignition(
|
|||||||
f.write(container_image.read())
|
f.write(container_image.read())
|
||||||
|
|
||||||
|
|
||||||
def build_fuelignition() -> docker.models.images.Image:
|
def build_fuelignition() -> docker.models.images.Image: # type: ignore
|
||||||
"""Builds the fuel-ignition docker image
|
"""Builds the fuel-ignition docker image
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
@@ -196,12 +199,14 @@ def json_to_img(
|
|||||||
fuelignition_container = None
|
fuelignition_container = None
|
||||||
fuelignition_image = None
|
fuelignition_image = None
|
||||||
try:
|
try:
|
||||||
|
driver_port = next_free_tcp_port(4444)
|
||||||
# Initialise containers
|
# Initialise containers
|
||||||
selenium_container = config.CLIENT.containers.run(
|
selenium_container = config.CLIENT.containers.run(
|
||||||
"selenium/standalone-firefox:latest",
|
"selenium/standalone-firefox:latest",
|
||||||
detach=True,
|
detach=True,
|
||||||
remove=True,
|
remove=True,
|
||||||
ports={4444: 4444, 7900: 7900},
|
network_mode="bridge",
|
||||||
|
ports={4444: driver_port},
|
||||||
mounts=[
|
mounts=[
|
||||||
config.CWD_MOUNT,
|
config.CWD_MOUNT,
|
||||||
],
|
],
|
||||||
@@ -224,7 +229,7 @@ def json_to_img(
|
|||||||
):
|
):
|
||||||
time.sleep(0.1)
|
time.sleep(0.1)
|
||||||
# Now, create the webdriver and convert the json to an img
|
# Now, create the webdriver and convert the json to an img
|
||||||
driver = create_driver()
|
driver = create_driver(driver_port)
|
||||||
convert_json_via_fuelignition(selenium_container, driver, json_path, img_path)
|
convert_json_via_fuelignition(selenium_container, driver, json_path, img_path)
|
||||||
driver.quit()
|
driver.quit()
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
|
|||||||
@@ -8,7 +8,14 @@ import tomllib
|
|||||||
|
|
||||||
CLIENT = docker.from_env(version="auto")
|
CLIENT = docker.from_env(version="auto")
|
||||||
MAX_PORT: int = 65535
|
MAX_PORT: int = 65535
|
||||||
PROJECT_ROOT: Path = Path(__file__).parent.parent.parent.absolute()
|
|
||||||
|
def __get_project_root():
|
||||||
|
r = Path(__file__)
|
||||||
|
while r.name != "src":
|
||||||
|
r = r.parent
|
||||||
|
return r.parent
|
||||||
|
|
||||||
|
PROJECT_ROOT: Path = __get_project_root()
|
||||||
|
|
||||||
ConfigLabel = Union[str, list[str]] # After PEP695 support: type ConfigLabel = str | list[str]
|
ConfigLabel = Union[str, list[str]] # After PEP695 support: type ConfigLabel = str | list[str]
|
||||||
|
|
||||||
@@ -63,19 +70,19 @@ class Config(SimpleNamespace):
|
|||||||
for k, v in config.items():
|
for k, v in config.items():
|
||||||
match k:
|
match k:
|
||||||
case "SRC_DIR" | "BUILD_DIR":
|
case "SRC_DIR" | "BUILD_DIR":
|
||||||
config[k] = Path(v).absolute()
|
config[k] = Path(PROJECT_ROOT / v).absolute()
|
||||||
case "CWD_MOUNTDIR":
|
case "CWD_MOUNTDIR":
|
||||||
config[k] = Path(v)
|
config[k] = Path(v)
|
||||||
# Then, get required paths from config or globals if not present
|
# Then, get required paths from config or globals if not present
|
||||||
build_dir = config.get("BUILD_DIR", self.BUILD_DIR)
|
build_dir = Path(config.get("BUILD_DIR", self.BUILD_DIR)).absolute()
|
||||||
cwd_mountdir = config.get("CWD_MOUNTDIR", self.CWD_MOUNTDIR)
|
cwd_mountdir = Path(config.get("CWD_MOUNTDIR", self.CWD_MOUNTDIR))
|
||||||
src_dir = config.get("SRC_DIR", self.SRC_DIR)
|
src_dir = Path(config.get("SRC_DIR", self.SRC_DIR)).absolute()
|
||||||
# Finally, construct the secondary parameters
|
# Finally, construct the secondary parameters
|
||||||
config["FUELIGNITION_BUILD_DIR"] = build_dir / config.get(
|
config["FUELIGNITION_BUILD_DIR"] = build_dir / config.get(
|
||||||
"FUELIGNITION_BUILD_DIR", self.FUELIGNITION_BUILD_DIR
|
"FUELIGNITION_BUILD_DIR", self.FUELIGNITION_BUILD_DIR
|
||||||
)
|
)
|
||||||
config["DOCKERFILE_DIR"] = src_dir / config.get("DOCKERFILE_DIR", self.DOCKERFILE_DIR)
|
config["DOCKERFILE_DIR"] = src_dir / config.get("DOCKERFILE_DIR", self.DOCKERFILE_DIR)
|
||||||
config["CWD_MOUNT"] = docker.types.Mount(
|
config["CWD_MOUNT"] = docker.types.Mount( # type: ignore <- I really wish docker-py had typeshed stubs
|
||||||
target=str(cwd_mountdir),
|
target=str(cwd_mountdir),
|
||||||
source=str(PROJECT_ROOT),
|
source=str(PROJECT_ROOT),
|
||||||
type="bind",
|
type="bind",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
from fnmatch import fnmatch
|
from fnmatch import fnmatch
|
||||||
import ipaddress
|
import ipaddress
|
||||||
from typing import Annotated, Optional, Union
|
from typing import Annotated, Optional
|
||||||
|
|
||||||
from docker.types import Mount
|
from docker.types import Mount
|
||||||
import typer
|
import typer
|
||||||
@@ -14,7 +14,7 @@ from .utils import ensure_build_dir
|
|||||||
|
|
||||||
# When PEP695 is supported this line should be:
|
# When PEP695 is supported this line should be:
|
||||||
# type IPAddress = ipaddress.IPv4Address | ipaddress.IPv6Address
|
# type IPAddress = ipaddress.IPv4Address | ipaddress.IPv6Address
|
||||||
IPAddress = Union[ipaddress.IPv4Address, ipaddress.IPv6Address]
|
IPAddress = ipaddress._IPAddressBase
|
||||||
|
|
||||||
|
|
||||||
def filter_validation_response(response: str) -> str:
|
def filter_validation_response(response: str) -> str:
|
||||||
@@ -89,6 +89,7 @@ def write_disk(disk: str) -> None:
|
|||||||
mounts=[config.CWD_MOUNT, Mount("/ignition_disk", disk, type="bind")],
|
mounts=[config.CWD_MOUNT, Mount("/ignition_disk", disk, type="bind")],
|
||||||
privileged=True,
|
privileged=True,
|
||||||
command=f"dd if={config.CWD_MOUNTDIR}/build/ignition.img of=/ignition_disk",
|
command=f"dd if={config.CWD_MOUNTDIR}/build/ignition.img of=/ignition_disk",
|
||||||
|
remove=config.CLEANUP_CONTAINERS,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import ipaddress
|
import ipaddress
|
||||||
import json
|
import json
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from typing import Annotated, Optional, Union
|
from typing import Annotated, Optional
|
||||||
|
|
||||||
import typer
|
import typer
|
||||||
|
|
||||||
@@ -13,7 +13,7 @@ from .utils import ensure_build_dir
|
|||||||
|
|
||||||
# When PEP695 is supported this line should be:
|
# When PEP695 is supported this line should be:
|
||||||
# type IPAddress = ipaddress.IPv4Address | ipaddress.IPv6Address
|
# type IPAddress = ipaddress.IPv4Address | ipaddress.IPv6Address
|
||||||
IPAddress = Union[ipaddress.IPv4Address, ipaddress.IPv6Address]
|
IPAddress = ipaddress._IPAddressBase
|
||||||
|
|
||||||
|
|
||||||
def load_template() -> dict:
|
def load_template() -> dict:
|
||||||
@@ -33,7 +33,7 @@ def apply_ignition_settings(
|
|||||||
password: str,
|
password: str,
|
||||||
swarm_config: str,
|
swarm_config: str,
|
||||||
) -> dict:
|
) -> dict:
|
||||||
"""Applies the specified ignition settings to the template
|
"""Applies the specified ignition settings to the given template
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
template (dict): The template to apply the settings to
|
template (dict): The template to apply the settings to
|
||||||
@@ -46,7 +46,11 @@ def apply_ignition_settings(
|
|||||||
"""
|
"""
|
||||||
ignition_config = template.copy()
|
ignition_config = template.copy()
|
||||||
ignition_config["hostname"] = hostname
|
ignition_config["hostname"] = hostname
|
||||||
|
if password:
|
||||||
ignition_config["login"]["users"][0]["passwd"] = password
|
ignition_config["login"]["users"][0]["passwd"] = password
|
||||||
|
ignition_config["login"]["users"][0]["hash_type"] = "bcrypt"
|
||||||
|
elif not config.TESTING:
|
||||||
|
raise ValueError("Password must be specified")
|
||||||
|
|
||||||
# Add files that will define a service to ensure that the node joins the swarm
|
# Add files that will define a service to ensure that the node joins the swarm
|
||||||
with open(config.SRC_DIR / "templates/join_swarm.sh", "r") as f1, open(
|
with open(config.SRC_DIR / "templates/join_swarm.sh", "r") as f1, open(
|
||||||
@@ -186,6 +190,12 @@ def create_img(
|
|||||||
Enable debug mode.
|
Enable debug mode.
|
||||||
Defaults to False.
|
Defaults to False.
|
||||||
"""
|
"""
|
||||||
|
# Guards against the user not specifying a password
|
||||||
|
if password is None and not config.TESTING:
|
||||||
|
raise typer.BadParameter("Password must be specified")
|
||||||
|
elif password is None:
|
||||||
|
password = ""
|
||||||
|
|
||||||
# get swarm configuration as JSON
|
# get swarm configuration as JSON
|
||||||
swarm_config = json.dumps(
|
swarm_config = json.dumps(
|
||||||
{
|
{
|
||||||
@@ -195,10 +205,6 @@ def create_img(
|
|||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
# Guards against the user not specifying a password
|
|
||||||
if password is None:
|
|
||||||
raise typer.BadParameter("Password must be specified")
|
|
||||||
|
|
||||||
# Create ignition configuration
|
# Create ignition configuration
|
||||||
ignition_config = apply_ignition_settings(
|
ignition_config = apply_ignition_settings(
|
||||||
load_template(),
|
load_template(),
|
||||||
|
|||||||
2
src/node_deployer/node_deployer.py
Executable file → Normal file
2
src/node_deployer/node_deployer.py
Executable file → Normal file
@@ -1,3 +1,5 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
|
||||||
from typing import Any, Dict
|
from typing import Any, Dict
|
||||||
|
|
||||||
import typer
|
import typer
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ def ensure_build_dir(f: Callable) -> Callable:
|
|||||||
Returns:
|
Returns:
|
||||||
Callable: The decorated function
|
Callable: The decorated function
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@wraps(f)
|
@wraps(f)
|
||||||
def wrapper(*args, **kwargs):
|
def wrapper(*args, **kwargs):
|
||||||
Path(config.BUILD_DIR).mkdir(exist_ok=True, parents=True)
|
Path(config.BUILD_DIR).mkdir(exist_ok=True, parents=True)
|
||||||
@@ -24,6 +25,7 @@ def ensure_build_dir(f: Callable) -> Callable:
|
|||||||
|
|
||||||
class Singleton(type):
|
class Singleton(type):
|
||||||
"""A singleton metaclass"""
|
"""A singleton metaclass"""
|
||||||
|
|
||||||
_instance = None
|
_instance = None
|
||||||
|
|
||||||
def __call__(cls, *args, **kwargs):
|
def __call__(cls, *args, **kwargs):
|
||||||
@@ -35,3 +37,32 @@ class Singleton(type):
|
|||||||
if cls._instance is None:
|
if cls._instance is None:
|
||||||
cls._instance = super().__call__(*args, **kwargs)
|
cls._instance = super().__call__(*args, **kwargs)
|
||||||
return cls._instance
|
return cls._instance
|
||||||
|
|
||||||
|
|
||||||
|
def next_free_tcp_port(port: int) -> int:
|
||||||
|
"""Finds the next free port after the specified port
|
||||||
|
|
||||||
|
Args:
|
||||||
|
port (int): The port to start searching from
|
||||||
|
|
||||||
|
Raises:
|
||||||
|
ValueError: If no free ports are found
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
int: The next free port
|
||||||
|
"""
|
||||||
|
containers = config.CLIENT.containers.list(all=True)
|
||||||
|
ports = []
|
||||||
|
for container in containers:
|
||||||
|
port_values = container.ports.values()
|
||||||
|
if not port_values:
|
||||||
|
continue
|
||||||
|
for x in list(container.ports.values())[0]:
|
||||||
|
ports.append(int(x["HostPort"]))
|
||||||
|
ports = set(ports)
|
||||||
|
while port in ports:
|
||||||
|
port += 1
|
||||||
|
if port > 65535:
|
||||||
|
raise ValueError("No free ports")
|
||||||
|
return port
|
||||||
|
|
||||||
@@ -2,8 +2,7 @@
|
|||||||
"login": {
|
"login": {
|
||||||
"users": [
|
"users": [
|
||||||
{
|
{
|
||||||
"name": "root",
|
"name": "root"
|
||||||
"hash_type": "bcrypt"
|
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
|||||||
0
tests/__init__.py
Normal file
0
tests/__init__.py
Normal file
5
tests/test_node_deployer/__init__.py
Normal file
5
tests/test_node_deployer/__init__.py
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
from node_deployer.config import config
|
||||||
|
|
||||||
|
|
||||||
|
config.update_config("test")
|
||||||
|
config.BUILD_DIR = config.BUILD_DIR / "tests"
|
||||||
76
tests/test_node_deployer/data/config.ign
Normal file
76
tests/test_node_deployer/data/config.ign
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
{
|
||||||
|
"ignition": {
|
||||||
|
"version": "3.2.0"
|
||||||
|
},
|
||||||
|
"passwd": {
|
||||||
|
"users": [
|
||||||
|
{
|
||||||
|
"name": "root",
|
||||||
|
"passwordHash": "$2a$08$YdxsvKFRc1q2S1lmVixm5Oel3Y2oCNAsx9Sh2Dx4pL/udApPzUQu6"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"storage": {
|
||||||
|
"files": [
|
||||||
|
{
|
||||||
|
"path": "/root/join_swarm.json",
|
||||||
|
"mode": 420,
|
||||||
|
"overwrite": true,
|
||||||
|
"contents": {
|
||||||
|
"source": "data:text/plain;charset=utf-8;base64,eyJTV0lUQ0hfSVBfQUREUkVTUyI6ICIxOTIuMTY4LjEuMSIsICJTV0lUQ0hfUE9SVCI6IDQyLCAiU1dBUk1fVE9LRU4iOiAiU1dNVEtOLTEtVEhJU0lTQVRFU1RTV0FSTVRPS0VORk9SVEVTVElOR1BVUlBPU0VTQU5EVEhBVE1FQU5TSVRORUVEU1RPQkVRVUlURUxPTkcifQ=="
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/root/join_swarm.sh",
|
||||||
|
"mode": 420,
|
||||||
|
"overwrite": true,
|
||||||
|
"contents": {
|
||||||
|
"source": "data:text/plain;charset=utf-8;base64,IyEvYmluL2Jhc2gKCmlmIFtbICRFVUlEIC1uZSAwIF1dOyB0aGVuCiAgIGVjaG8gIlRoaXMgc2NyaXB0IG11c3QgYmUgcnVuIGFzIHJvb3QiIAogICBleGl0IDEKZmkKCiMgTG9hZCB0aGUgY29uZmlnIGZpbGUgaW50byB2YXJpYWJsZXMKZXZhbCAiJChqcSAtciAndG9fZW50cmllc1tdIHwgImV4cG9ydCBcKC5rZXkpPVwoLnZhbHVlIHwgQHNoKSInIC9yb290L2pvaW5fc3dhcm0uanNvbikiCgppZiBbWyAkKGRvY2tlciBpbmZvIHwgZ3JlcCBTd2FybSB8IGF3ayAne3ByaW50ICQyfScpID09ICJpbmFjdGl2ZSIgXV07IHRoZW4KICAgIGRvY2tlciBzd2FybSBqb2luIC0tdG9rZW4gJFNXQVJNX1RPS0VOIFskU1dJVENIX0lQX0FERFJFU1NdOiRTV0lUQ0hfUE9SVAplbHNlCiAgICBlY2hvICJUaGlzIG5vZGUgaXMgYWxyZWFkeSBwYXJ0IG9mIGEgc3dhcm0iCiAgICBkb2NrZXIgaW5mbyAtZiBqc29uIHwganEgLlN3YXJtCmZpCg=="
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/etc/hostname",
|
||||||
|
"mode": 420,
|
||||||
|
"overwrite": true,
|
||||||
|
"contents": {
|
||||||
|
"source": "data:,test_hostname"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/etc/NetworkManager/system-connections/eth0.nmconnection",
|
||||||
|
"mode": 384,
|
||||||
|
"overwrite": true,
|
||||||
|
"contents": {
|
||||||
|
"source": "data:text/plain;charset=utf-8;base64,Cltjb25uZWN0aW9uXQppZD1ldGgwCnR5cGU9ZXRoZXJuZXQKaW50ZXJmYWNlLW5hbWU9ZXRoMAoKW2lwdjRdCmRucy1zZWFyY2g9Cm1ldGhvZD1hdXRvCgpbaXB2Nl0KZG5zLXNlYXJjaD0KYWRkci1nZW4tbW9kZT1ldWk2NAptZXRob2Q9YXV0bwo=",
|
||||||
|
"human_read": "\n[connection]\nid=eth0\ntype=ethernet\ninterface-name=eth0\n\n[ipv4]\ndns-search=\nmethod=auto\n\n[ipv6]\ndns-search=\naddr-gen-mode=eui64\nmethod=auto\n"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/etc/NetworkManager/conf.d/noauto.conf",
|
||||||
|
"mode": 420,
|
||||||
|
"overwrite": true,
|
||||||
|
"contents": {
|
||||||
|
"source": "data:text/plain;charset=utf-8;base64,W21haW5dCiMgRG8gbm90IGRvIGF1dG9tYXRpYyAoREhDUC9TTEFBQykgY29uZmlndXJhdGlvbiBvbiBldGhlcm5ldCBkZXZpY2VzCiMgd2l0aCBubyBvdGhlciBtYXRjaGluZyBjb25uZWN0aW9ucy4Kbm8tYXV0by1kZWZhdWx0PSoK",
|
||||||
|
"human_read": "[main]\n# Do not do automatic (DHCP/SLAAC) configuration on ethernet devices\n# with no other matching connections.\nno-auto-default=*\n"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"systemd": {
|
||||||
|
"units": [
|
||||||
|
{
|
||||||
|
"name": "cockpit.socket.service",
|
||||||
|
"enabled": true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "docker.service",
|
||||||
|
"enabled": true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "join_swarm.service",
|
||||||
|
"enabled": false,
|
||||||
|
"contents": "[Unit]\nDescription=Ensure that node joins a swarm on startup\n\n[Service]\nExecStart=/root/join_swarm.sh\n\n[Install]\nWantedBy=multi-user.target"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
Binary file not shown.
BIN
tests/test_node_deployer/data/create_img/load_template.pkl
Normal file
BIN
tests/test_node_deployer/data/create_img/load_template.pkl
Normal file
Binary file not shown.
65
tests/test_node_deployer/data/fuelignition.json
Normal file
65
tests/test_node_deployer/data/fuelignition.json
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
{
|
||||||
|
"login": {
|
||||||
|
"users": [
|
||||||
|
{
|
||||||
|
"name": "root"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"network": {
|
||||||
|
"interfaces": [
|
||||||
|
{
|
||||||
|
"name": "eth0",
|
||||||
|
"ipv4": {
|
||||||
|
"network_type": "DHCP",
|
||||||
|
"auto_dns_enabled": true
|
||||||
|
},
|
||||||
|
"ipv6": {
|
||||||
|
"network_type": "DHCP",
|
||||||
|
"auto_dns_enabled": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"systemd": {
|
||||||
|
"units": [
|
||||||
|
{
|
||||||
|
"name": "cockpit.socket.service",
|
||||||
|
"enabled": "yes"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "docker.service",
|
||||||
|
"enabled": "yes"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "join_swarm.service",
|
||||||
|
"enabled": true,
|
||||||
|
"contents": "[Unit]\nDescription=Ensure that node joins a swarm on startup\n\n[Service]\nExecStart=/root/join_swarm.sh\n\n[Install]\nWantedBy=multi-user.target"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"package": {
|
||||||
|
"install": [
|
||||||
|
"patterns-microos-cockpit, docker, jq"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"hostname": "test_hostname",
|
||||||
|
"storage": {
|
||||||
|
"files": [
|
||||||
|
{
|
||||||
|
"path": "/root/join_swarm.json",
|
||||||
|
"source_type": "data",
|
||||||
|
"mode": 420,
|
||||||
|
"overwrite": true,
|
||||||
|
"data_content": "{\"SWITCH_IP_ADDRESS\": \"192.168.1.1\", \"SWITCH_PORT\": 42, \"SWARM_TOKEN\": \"SWMTKN-1-THISISATESTSWARMTOKENFORTESTINGPURPOSESANDTHATMEANSITNEEDSTOBEQUITELONG\"}"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/root/join_swarm.sh",
|
||||||
|
"source_type": "data",
|
||||||
|
"mode": 420,
|
||||||
|
"overwrite": true,
|
||||||
|
"data_content": "#!/bin/bash\n\nif [[ $EUID -ne 0 ]]; then\n echo \"This script must be run as root\" \n exit 1\nfi\n\n# Load the config file into variables\neval \"$(jq -r 'to_entries[] | \"export \\(.key)=\\(.value | @sh)\"' /root/join_swarm.json)\"\n\nif [[ $(docker info | grep Swarm | awk '{print $2}') == \"inactive\" ]]; then\n docker swarm join --token $SWARM_TOKEN [$SWITCH_IP_ADDRESS]:$SWITCH_PORT\nelse\n echo \"This node is already part of a swarm\"\n docker info -f json | jq .Swarm\nfi\n"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
BIN
tests/test_node_deployer/data/ignition.img
Normal file
BIN
tests/test_node_deployer/data/ignition.img
Normal file
Binary file not shown.
16
tests/test_node_deployer/test_autoignition.py
Normal file
16
tests/test_node_deployer/test_autoignition.py
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
import filecmp
|
||||||
|
|
||||||
|
from node_deployer import autoignition
|
||||||
|
from node_deployer.config import config
|
||||||
|
|
||||||
|
|
||||||
|
class TestAutoignition:
|
||||||
|
def test_json_to_img(self, tmpdir):
|
||||||
|
autoignition.json_to_img(
|
||||||
|
config.PROJECT_ROOT / "tests/test_node_deployer/data/fuelignition.json",
|
||||||
|
tmpdir / "ignition.img",
|
||||||
|
)
|
||||||
|
assert filecmp.cmp(
|
||||||
|
config.PROJECT_ROOT / "tests/test_node_deployer/data/ignition.img",
|
||||||
|
tmpdir / "ignition.img",
|
||||||
|
)
|
||||||
50
tests/test_node_deployer/test_create_img.py
Normal file
50
tests/test_node_deployer/test_create_img.py
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
import filecmp
|
||||||
|
import pickle
|
||||||
|
|
||||||
|
from node_deployer import create_img
|
||||||
|
from node_deployer.config import config
|
||||||
|
|
||||||
|
|
||||||
|
class TestCreateImg:
|
||||||
|
def test_load_template(self):
|
||||||
|
template = create_img.load_template()
|
||||||
|
with open(
|
||||||
|
config.PROJECT_ROOT / "tests/test_node_deployer/data/create_img/load_template.pkl", "rb"
|
||||||
|
) as f:
|
||||||
|
assert pickle.load(f) == template
|
||||||
|
|
||||||
|
def test_apply_ignition_settings(self):
|
||||||
|
with open(
|
||||||
|
config.PROJECT_ROOT / "tests/test_node_deployer/data/create_img/load_template.pkl",
|
||||||
|
mode="rb",
|
||||||
|
) as f:
|
||||||
|
template = pickle.load(f)
|
||||||
|
test_result = create_img.apply_ignition_settings(
|
||||||
|
template,
|
||||||
|
"test_hostname",
|
||||||
|
"",
|
||||||
|
{
|
||||||
|
"SWITCH_IP_ADDRESS": "192.168.1.1",
|
||||||
|
"SWITCH_PORT": 42,
|
||||||
|
"SWARM_TOKEN": "SWMTKN-1-THISISATESTSWARMTOKENFORTESTINGPURPOSESANDTHATMEANSITNEEDSTOBEQUITELONG", # noqa: E501
|
||||||
|
},
|
||||||
|
)
|
||||||
|
with open(
|
||||||
|
config.PROJECT_ROOT
|
||||||
|
/ "tests/test_node_deployer/data/create_img/apply_ignition_settings.pkl",
|
||||||
|
mode="rb",
|
||||||
|
) as f:
|
||||||
|
assert pickle.load(f) == test_result
|
||||||
|
|
||||||
|
def test_create_img(self, tmpdir):
|
||||||
|
create_img.create_img(
|
||||||
|
hostname="test_hostname",
|
||||||
|
password="",
|
||||||
|
switch_ip="192.168.1.1",
|
||||||
|
switch_port=42,
|
||||||
|
img_path=tmpdir / "ignition.img",
|
||||||
|
)
|
||||||
|
assert filecmp.cmp(
|
||||||
|
tmpdir / "ignition.img",
|
||||||
|
config.PROJECT_ROOT / "tests/test_node_deployer/data/ignition.img",
|
||||||
|
)
|
||||||
Reference in New Issue
Block a user