initial code towards services that will generate files from templates and provide configuration
This commit is contained in:
parent
5490f961a6
commit
988ca76cec
8 changed files with 346 additions and 129 deletions
0
daemon/core/configservice/__init__.py
Normal file
0
daemon/core/configservice/__init__.py
Normal file
128
daemon/core/configservice/base.py
Normal file
128
daemon/core/configservice/base.py
Normal file
|
@ -0,0 +1,128 @@
|
|||
import abc
|
||||
import enum
|
||||
import inspect
|
||||
import logging
|
||||
import pathlib
|
||||
import time
|
||||
from typing import Any, Dict, List
|
||||
|
||||
from mako import exceptions
|
||||
from mako.lookup import TemplateLookup
|
||||
|
||||
from core.errors import CoreCommandError, CoreError
|
||||
from core.nodes.base import CoreNode
|
||||
|
||||
TEMPLATES_DIR = "templates"
|
||||
|
||||
|
||||
class ConfigServiceMode(enum.Enum):
|
||||
BLOCKING = 0
|
||||
NON_BLOCKING = 1
|
||||
TIMER = 2
|
||||
|
||||
|
||||
class ConfigService(abc.ABC):
|
||||
# validation period in seconds, how frequent validation is attempted
|
||||
validation_period = 0.5
|
||||
|
||||
# time to wait in seconds for determining if service started successfully
|
||||
validation_timer = 5
|
||||
|
||||
def __init__(self, node: CoreNode) -> None:
|
||||
self.node = node
|
||||
class_file = inspect.getfile(self.__class__)
|
||||
templates_path = pathlib.Path(class_file).parent.joinpath(TEMPLATES_DIR)
|
||||
logging.info(templates_path)
|
||||
self.templates = TemplateLookup(directories=templates_path)
|
||||
|
||||
@property
|
||||
@abc.abstractmethod
|
||||
def name(self):
|
||||
raise NotImplementedError
|
||||
|
||||
@property
|
||||
@abc.abstractmethod
|
||||
def group(self):
|
||||
raise NotImplementedError
|
||||
|
||||
@property
|
||||
@abc.abstractmethod
|
||||
def executables(self) -> List[str]:
|
||||
raise NotImplementedError
|
||||
|
||||
@property
|
||||
@abc.abstractmethod
|
||||
def dependencies(self) -> List[str]:
|
||||
raise NotImplementedError
|
||||
|
||||
@property
|
||||
@abc.abstractmethod
|
||||
def startup(self) -> List[str]:
|
||||
raise NotImplementedError
|
||||
|
||||
@property
|
||||
@abc.abstractmethod
|
||||
def validate(self) -> List[str]:
|
||||
raise NotImplementedError
|
||||
|
||||
@property
|
||||
@abc.abstractmethod
|
||||
def shutdown(self) -> List[str]:
|
||||
raise NotImplementedError
|
||||
|
||||
@property
|
||||
@abc.abstractmethod
|
||||
def validation_mode(self) -> ConfigServiceMode:
|
||||
raise NotImplementedError
|
||||
|
||||
def start(self) -> None:
|
||||
if not self.startup:
|
||||
return
|
||||
wait = self.validation_mode == ConfigServiceMode.BLOCKING
|
||||
start = time.monotonic()
|
||||
index = 0
|
||||
cmds = self.startup[:]
|
||||
while cmds:
|
||||
cmd = cmds[index]
|
||||
try:
|
||||
self.node.cmd(cmd, wait=wait)
|
||||
del cmds[index]
|
||||
index += 1
|
||||
except CoreCommandError:
|
||||
logging.exception("error starting command")
|
||||
time.sleep(self.validation_period)
|
||||
|
||||
if time.monotonic() - start > 0:
|
||||
raise CoreError(
|
||||
f"node({self.node.name}) service({self.name()}) failed to start"
|
||||
)
|
||||
|
||||
def stop(self) -> None:
|
||||
if not self.shutdown:
|
||||
return
|
||||
for cmd in self.shutdown:
|
||||
self.node.cmd(cmd, wait=False)
|
||||
|
||||
def restart(self):
|
||||
self.stop()
|
||||
self.start()
|
||||
|
||||
def run(self, cmd: str, wait: bool = True):
|
||||
self.node.cmd(cmd, wait)
|
||||
|
||||
def create_files(self) -> None:
|
||||
raise NotImplementedError
|
||||
|
||||
def render(self, name: str, data: Dict[str, Any] = None) -> None:
|
||||
if data is None:
|
||||
data = {}
|
||||
try:
|
||||
template = self.templates.get_template(name)
|
||||
rendered = template.render_unicode(node=self.node, **data)
|
||||
print(rendered)
|
||||
# self.node.nodefile(name, rendered)
|
||||
except Exception:
|
||||
raise CoreError(
|
||||
f"error rendering template: {name}"
|
||||
f"{exceptions.text_error_template().render()}"
|
||||
)
|
0
daemon/core/configservices/__init__.py
Normal file
0
daemon/core/configservices/__init__.py
Normal file
0
daemon/core/configservices/defaultroute/__init__.py
Normal file
0
daemon/core/configservices/defaultroute/__init__.py
Normal file
47
daemon/core/configservices/defaultroute/service.py
Normal file
47
daemon/core/configservices/defaultroute/service.py
Normal file
|
@ -0,0 +1,47 @@
|
|||
import logging
|
||||
|
||||
import netaddr
|
||||
|
||||
from core.configservice.base import ConfigService, ConfigServiceMode
|
||||
from core.emulator.session import Session
|
||||
from core.nodes.base import CoreNode
|
||||
from core.nodes.interface import Veth
|
||||
|
||||
|
||||
class DefaultRoute(ConfigService):
|
||||
name = "DefaultRoute"
|
||||
group = "Utility"
|
||||
executables = []
|
||||
dependencies = []
|
||||
startup = []
|
||||
validate = []
|
||||
shutdown = []
|
||||
validation_mode = ConfigServiceMode.BLOCKING
|
||||
|
||||
def create_files(self):
|
||||
self.create_default_route()
|
||||
|
||||
def create_default_route(self):
|
||||
addresses = []
|
||||
for netif in self.node.netifs():
|
||||
if getattr(netif, "control", False):
|
||||
continue
|
||||
for addr in netif.addrlist:
|
||||
net = netaddr.IPNetwork(addr)
|
||||
if net[1] != net[-2]:
|
||||
addresses.append(net[1])
|
||||
data = dict(addresses=addresses)
|
||||
self.render("defaultroute.sh", data)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
logging.basicConfig(level=logging.INFO)
|
||||
session = Session(1, mkdir=False)
|
||||
node = CoreNode(session, _id=1, start=False)
|
||||
netif = Veth(session, node, "eth0", "eth0", start=False)
|
||||
netif.addaddr("10.0.0.1/24")
|
||||
node.addnetif(netif, 0)
|
||||
service = DefaultRoute(node)
|
||||
service.create_files()
|
||||
# data = service.render(node, "defaultroute.sh", dict(addresses=[]))
|
||||
# print(data)
|
|
@ -0,0 +1,5 @@
|
|||
#!/bin/sh
|
||||
# auto-generated by DefaultRoute service
|
||||
% for address in addresses:
|
||||
ip route add default via ${address}
|
||||
% endfor
|
Loading…
Add table
Add a link
Reference in a new issue