Skip to content

Commit

Permalink
Controller updates from gpantelakis
Browse files Browse the repository at this point in the history
Updates patch that fix Controller and cli to better handle setup.

Follow up changes per PR review also added to cli_commands.py.

Adding --no-conf option to allow running without a config file.

Add note about where NaturalOrderGroup in cli_commands came from.
  • Loading branch information
spoore1 committed Dec 6, 2024
1 parent d998bf3 commit 5f2cfab
Show file tree
Hide file tree
Showing 3 changed files with 106 additions and 64 deletions.
87 changes: 54 additions & 33 deletions SCAutolib/cli_commands.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,9 @@
def check_conf_path(conf):
return click.Path(exists=True, resolve_path=True)(conf)


# In Help output, force the subcommand list to match the order
# listed in this file. Solution was found here:
# https://github.com/pallets/click/issues/513#issuecomment-301046782
class NaturalOrderGroup(click.Group):
"""
Command group trying to list subcommands in the order they were added.
Expand All @@ -34,8 +36,8 @@ def __init__(self, name=None, commands=None, **attrs):
elif not isinstance(commands, OrderedDict):
commands = OrderedDict(commands)
click.Group.__init__(self, name=name,
commands=commands,
**attrs)
commands=commands,
**attrs)

def list_commands(self, ctx):
"""
Expand All @@ -52,6 +54,8 @@ def list_commands(self, ctx):
default="./conf.json",
show_default=True,
help="Path to JSON configuration file.")
@click.option("--no-conf", is_flag=True, default=False, show_default=True,
help="Skip reading config file for manual gui test mode.")
@click.option('--force', "-f", is_flag=True, default=False, show_default=True,
help="Force the command to overwrite configuration if it exists.")
@click.option("--verbose", "-v", default="DEBUG", show_default=True,
Expand All @@ -60,12 +64,16 @@ def list_commands(self, ctx):
case_sensitive=False),
help="Verbosity level.")
@click.pass_context
def cli(ctx, force, verbose, conf):
def cli(ctx, force, verbose, conf, no_conf):
logger.setLevel(verbose)
ctx.ensure_object(dict) # Create a dict to store the context
ctx.obj["FORCE"] = force # Store the force option in the context
if ctx.invoked_subcommand and not gui:
ctx.obj["CONTROLLER"] = Controller(check_conf_path(conf))
ctx.obj["NO_CONF"] = no_conf
parsed_conf = None
if ctx.invoked_subcommand != gui:
parsed_conf = check_conf_path(conf)
if not no_conf:
ctx.obj["CONTROLLER"] = Controller(parsed_conf)


@cli.command()
Expand Down Expand Up @@ -202,8 +210,13 @@ def cleanup(ctx):


@cli.group(cls=NaturalOrderGroup, chain=True)
@click.option("--install-missing", "-i",
required=False,
default=False,
is_flag=True,
help="Install missing packages")
@click.pass_context
def gui(ctx):
def gui(ctx, install_missing):
""" Run GUI Test commands """
pass

Expand All @@ -215,19 +228,19 @@ def init():


@gui.command()
@click.option("--no",
required=False,
default=False,
is_flag=True,
help="Reverse the action")
@click.argument("name")
def assert_text(name):
def assert_text(name, no):
""" Check if a word is found on the screen """
if no:
return f"assert_no_text:{name}"
return f"assert_text:{name}"


@gui.command()
@click.argument("name")
def assert_no_text(name):
""" Check that a word is not found on the screen """
return f"assert_no_text:{name}"


@gui.command()
@click.argument("name")
def click_on(name):
Expand All @@ -236,9 +249,16 @@ def click_on(name):


@gui.command()
def check_home_screen():
@click.option("--no",
required=False,
default=False,
is_flag=True,
help="Reverse the action")
def check_home_screen(no):
""" Check if screen appears to be the home screen """
return f"check_home_screen"
if no:
return "check_no_home_screen"
return "check_home_screen"


@gui.command()
Expand All @@ -263,39 +283,40 @@ def done():

@gui.result_callback()
@click.pass_context
def run_all(ctx, actions):
click.echo(actions)
def run_all(ctx, actions, install_missing):
""" Run all cli actions in order """
if not ctx.obj["NO_CONF"]:
ctx.obj["CONTROLLER"].setup_graphical(install_missing, True)

from SCAutolib.models.gui import GUI
gui = GUI()
for action in actions:
if "init" in action:
gui.__enter__()
if "assert_text" in action:
assert_text = action.split(":")[1]
click.echo(f"CLICK: assert_text:{assert_text}")
assert_text = action.split(":", 1)[1]
gui.assert_text(assert_text)
if "assert_no_text" in action:
assert_text = action.split(":")[1]
click.echo(f"CLICK: assert_text:{assert_text}")
gui.assert_text(assert_text)
assert_text = action.split(":", 1)[1]
gui.assert_no_text(assert_text)
if "click_on" in action:
click_on = action.split(":")[1]
click.echo(f"CLICK: click_on:{click_on}")
click_on = action.split(":", 1)[1]
gui.click_on(click_on)
if "check_home_screen" in action:
click.echo("CLICK: check_home_screen")
gui.check_home_screen()
if "check_no_home_screen" in action:
gui.check_home_screen(False)
if "kb_send" in action:
params = action.split(":")[1].split()
click.echo(f"CLICK: kb_send:{params}")
gui.kb_send(*params)
params = action.split(":", 1)[1].split()[0]
gui.kb_send(params)
if "kb_write" in action:
params = action.split(":")[1].split()
click.echo(f"CLICK: kb_write:{params}")
gui.kb_write(*params)
params = action.split(":", 1)[1].split()[0]
gui.kb_write(params)
gui.kb_send('enter')
if "done" in action:
gui.__exit__(None, None, None)
if not ctx.obj["NO_CONF"]:
ctx.obj["CONTROLLER"].cleanup()


if __name__ == "__main__":
Expand Down
77 changes: 46 additions & 31 deletions SCAutolib/controller.py
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@ class Controller:
def conf_path(self):
return self._lib_conf_path

def __init__(self, config: Union[Path, str], params: {} = None):
def __init__(self, config: Union[Path, str] = None, params: {} = None):
"""
Constructor will parse and check input configuration file. If some
required fields in the configuration are missing, CLI parameters
Expand All @@ -55,15 +55,18 @@ def __init__(self, config: Union[Path, str], params: {} = None):
# Check params

# Parse config file
self._lib_conf_path = config.absolute() if isinstance(config, Path) \
else Path(config).absolute()

with self._lib_conf_path.open("r") as f:
tmp_conf = json.load(f)
if tmp_conf is None:
raise exceptions.SCAutolibException(
"Data are not loaded correctly.")
self.lib_conf = self._validate_configuration(tmp_conf, params)
self.lib_conf = None
if config:
self._lib_conf_path = config.absolute() if isinstance(config, Path) \
else Path(config).absolute()

with self._lib_conf_path.open("r") as f:
tmp_conf = json.load(f)
if tmp_conf is None:
raise exceptions.SCAutolibException(
"Data are not loaded correctly.")
self.lib_conf = self._validate_configuration(tmp_conf, params)

self.users = []
for d in (LIB_DIR, LIB_BACKUP, LIB_DUMP, LIB_DUMP_USERS, LIB_DUMP_CAS,
LIB_DUMP_CARDS, LIB_DUMP_CONFS):
Expand Down Expand Up @@ -147,12 +150,6 @@ def setup_system(self, install_missing: bool, gdm: bool, graphical: bool):

packages = ["opensc", "httpd", "sssd", "sssd-tools", "gnutls-utils",
"openssl", "nss-tools"]
if gdm:
packages.append("gdm")

if graphical:
# ffmpeg-free is in EPEL repo
packages += ["tesseract", "ffmpeg-free"]

# Prepare for virtual cards
if any(c["card_type"] == CardType.virtual
Expand Down Expand Up @@ -181,21 +178,7 @@ def setup_system(self, install_missing: bool, gdm: bool, graphical: bool):
raise exceptions.SCAutolibException(msg)

if graphical:
if not isDistro('fedora'):
run(['dnf', 'groupinstall', 'Server with GUI', '-y',
'--allowerasing'])
run(['pip', 'install', 'python-uinput'])
else:
# Fedora doesn't have server with GUI group so installed gdm
# manually and also python3-uinput should be installed from RPM
run(['dnf', 'install', 'gdm', 'python3-uinput', '-y'])
# disable subscription message
run(['systemctl', '--global', 'mask',
'org.gnome.SettingsDaemon.Subscription.target'])
# disable welcome message
self.dconf_file.create()
self.dconf_file.save()
run('dconf update')
self.setup_graphical(install_missing, gdm)

if not isDistro('fedora'):
run(['dnf', 'groupinstall', "Smart Card Support", '-y',
Expand All @@ -216,6 +199,38 @@ def setup_system(self, install_missing: bool, gdm: bool, graphical: bool):
dump_to_json(user.User(username="root",
password=self.lib_conf["root_passwd"]))

def setup_graphical(self, install_missing: bool, gdm: bool):
packages = ["gcc", "tesseract", "ffmpeg-free"]

if gdm:
packages.append("gdm")

missing = _check_packages(packages)
if install_missing and missing:
_install_packages(missing)
elif missing:
msg = "Can't continue with graphical. Some packages are missing: " \
f"{', '.join(missing)}"
logger.critical(msg)
raise exceptions.SCAutolibException(msg)

if not isDistro('fedora'):
run(['dnf', 'groupinstall', 'Server with GUI', '-y',
'--allowerasing'])
run(['pip', 'install', 'python-uinput'])
else:
# Fedora doesn't have server with GUI group so installed gdm
# manually and also python3-uinput should be installed from RPM
run(['dnf', 'install', 'gdm', 'python3-uinput', '-y'])
# disable subscription message
run(['systemctl', '--global', 'mask',
'org.gnome.SettingsDaemon.Subscription.target'])
# disable welcome message
if not self.dconf_file.exists():
self.dconf_file.create()
self.dconf_file.save()
run('dconf update')

def setup_local_ca(self, force: bool = False):
"""
Setup local CA based on configuration from the configuration file. All
Expand Down
6 changes: 6 additions & 0 deletions SCAutolib/models/file.py
Original file line number Diff line number Diff line change
Expand Up @@ -92,6 +92,12 @@ def remove(self):
f"Removed file {self._conf_file}."
)

def exists(self):
"""
Checks if a file exists. Returns boolean.
"""
return self._conf_file.exists()

def set(self, key: str, value: Union[int, str, bool], section: str = None,
separator: str = "="):
"""
Expand Down

0 comments on commit 5f2cfab

Please sign in to comment.