-
Notifications
You must be signed in to change notification settings - Fork 12
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: adding init command to entrypoints (#326)
* feat: make git arguments optional to support entrypoints that do not interact with git * feat: adding init command entrypoint * feat: adds logic to create provider cicd files for github Signed-off-by: George Vauter <gvauter@redhat.com> * feat: adding unit tests for init command Signed-off-by: George Vauter <gvauter@redhat.com> * fix: add license header and module docstring Signed-off-by: George Vauter <gvauter@redhat.com> * feat: add templates directory Signed-off-by: George Vauter <gvauter@redhat.com> * feat: add init command to poetry scripts Signed-off-by: George Vauter <gvauter@redhat.com> * fix: updating workflow templates Signed-off-by: George Vauter <gvauter@redhat.com> * fix: revert entrypoint_base and do not subclass init entrypoint Signed-off-by: George Vauter <gvauter@redhat.com> * fix: lists should be sorted before assertion Signed-off-by: George Vauter <gvauter@redhat.com> * fix: use importlib_resources backport for compatability with older python versions Signed-off-by: George Vauter <gvauter@redhat.com> * fix: run workflows on push instead of pull_request Signed-off-by: George Vauter <gvauter@redhat.com> * fix: check return code from call to compliance-trestle Signed-off-by: George Vauter <gvauter@redhat.com> --------- Signed-off-by: George Vauter <gvauter@redhat.com>
- Loading branch information
Showing
13 changed files
with
580 additions
and
7 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
{!CONTRIBUTING.md!} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
{!TROUBLESHOOTING.md!} |
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,174 @@ | ||
# SPDX-License-Identifier: Apache-2.0 | ||
# Copyright (c) 2024 Red Hat, Inc. | ||
|
||
"""Test for Init CLI entrypoint""" | ||
import argparse | ||
import logging | ||
import pathlib | ||
from typing import Dict | ||
from unittest.mock import Mock, patch | ||
|
||
import pytest | ||
from trestle.common.const import TRESTLE_CONFIG_DIR, TRESTLE_KEEP_FILE | ||
from trestle.common.file_utils import is_hidden | ||
|
||
from tests.testutils import args_dict_to_list, configure_test_logger, setup_for_init | ||
from trestlebot.const import GITHUB, TRESTLEBOT_CONFIG_DIR, TRESTLEBOT_KEEP_FILE | ||
from trestlebot.entrypoints.init import InitEntrypoint | ||
from trestlebot.entrypoints.init import main as cli_main | ||
from trestlebot.tasks.authored import types as model_types | ||
|
||
|
||
OSCAL_MODEL_SSP = model_types.AuthoredType.SSP.value | ||
OSCAL_MODEL_COMPDEF = model_types.AuthoredType.COMPDEF.value | ||
|
||
|
||
@pytest.fixture | ||
def args_dict() -> Dict[str, str]: | ||
return { | ||
"working-dir": ".", | ||
"provider": GITHUB, | ||
"oscal-model": OSCAL_MODEL_COMPDEF, | ||
} | ||
|
||
|
||
@patch( | ||
"trestlebot.entrypoints.log.configure_logger", | ||
Mock(side_effect=configure_test_logger), | ||
) | ||
def test_init_fails_if_trestlebot_dir_exists( | ||
tmp_init_dir: str, args_dict: Dict[str, str], caplog: pytest.LogCaptureFixture | ||
) -> None: | ||
"""Trestlebot init should fail if .trestlebot directory already exists""" | ||
setup_for_init(pathlib.Path(tmp_init_dir)) | ||
|
||
args_dict["working-dir"] = tmp_init_dir | ||
|
||
# Manulaly create .trestlebot dir so it already exists | ||
trestlebot_dir = pathlib.Path(tmp_init_dir) / pathlib.Path(TRESTLEBOT_CONFIG_DIR) | ||
trestlebot_dir.mkdir() | ||
|
||
with patch("sys.argv", ["trestlebot", *args_dict_to_list(args_dict)]): | ||
with pytest.raises(SystemExit, match="1"): | ||
cli_main() | ||
|
||
assert any( | ||
record.levelno == logging.ERROR | ||
and f"Initialization failed. Found existing {TRESTLEBOT_CONFIG_DIR} directory in" | ||
in record.message | ||
for record in caplog.records | ||
) | ||
|
||
|
||
@patch( | ||
"trestlebot.entrypoints.log.configure_logger", | ||
Mock(side_effect=configure_test_logger), | ||
) | ||
def test_init_if_not_git_repo( | ||
tmp_init_dir: str, args_dict: Dict[str, str], caplog: pytest.LogCaptureFixture | ||
) -> None: | ||
"""Test init fails if not in git repo directory""" | ||
args_dict["working-dir"] = tmp_init_dir | ||
with patch("sys.argv", ["trestlebot", *args_dict_to_list(args_dict)]): | ||
with pytest.raises(SystemExit, match="1"): | ||
cli_main() | ||
|
||
assert any( | ||
record.levelno == logging.ERROR | ||
and f"Initialization failed. Given directory {tmp_init_dir} is not a Git repository." | ||
in record.message | ||
for record in caplog.records | ||
) | ||
|
||
|
||
@patch( | ||
"trestlebot.entrypoints.log.configure_logger", | ||
Mock(side_effect=configure_test_logger), | ||
) | ||
def test_init_ssp_github( | ||
tmp_init_dir: str, args_dict: Dict[str, str], caplog: pytest.LogCaptureFixture | ||
) -> None: | ||
"""Tests for expected init command directories and files""" | ||
args_dict["working-dir"] = tmp_init_dir | ||
args_dict["oscal-model"] = OSCAL_MODEL_SSP | ||
args_dict["provider"] = GITHUB | ||
setup_for_init(pathlib.Path(tmp_init_dir)) | ||
with patch("sys.argv", ["trestlebot", *args_dict_to_list(args_dict)]): | ||
with pytest.raises(SystemExit, match="0"): | ||
cli_main() | ||
|
||
# .keep file should exist in .trestlebot repo | ||
tmp_dir = pathlib.Path(tmp_init_dir) | ||
trestlebot_dir = tmp_dir / pathlib.Path(TRESTLEBOT_CONFIG_DIR) | ||
keep_file = trestlebot_dir / pathlib.Path(TRESTLEBOT_KEEP_FILE) | ||
assert keep_file.exists() is True | ||
|
||
# directories for ssp model should exist | ||
model_dirs = [d.name for d in tmp_dir.iterdir() if not is_hidden(d)] | ||
expected = InitEntrypoint.MODEL_DIRS[args_dict["oscal-model"]] | ||
assert sorted(model_dirs) == sorted(expected) | ||
|
||
# directories for github workflows should exist | ||
workflow_dir = tmp_dir.joinpath(".github/workflows") | ||
workflow_files = [f.name for f in workflow_dir.iterdir()] | ||
expected = InitEntrypoint.PROVIDER_TEMPLATES[args_dict["provider"]][ | ||
args_dict["oscal-model"] | ||
] | ||
assert sorted(workflow_files) == sorted(expected) | ||
|
||
assert any( | ||
record.levelno == logging.INFO | ||
and f"Initialized trestlebot project successfully in {tmp_init_dir}" | ||
in record.message | ||
for record in caplog.records | ||
) | ||
|
||
|
||
@patch( | ||
"trestlebot.entrypoints.log.configure_logger", | ||
Mock(side_effect=configure_test_logger), | ||
) | ||
def test_init_compdef_github( | ||
tmp_init_dir: str, args_dict: Dict[str, str], caplog: pytest.LogCaptureFixture | ||
) -> None: | ||
"""Tests for expected init command directories and files""" | ||
args_dict["working-dir"] = tmp_init_dir | ||
args_dict["oscal-model"] = model_types.AuthoredType.COMPDEF.value | ||
args_dict["provider"] = GITHUB | ||
setup_for_init(pathlib.Path(tmp_init_dir)) | ||
|
||
with patch("sys.argv", ["trestlebot", *args_dict_to_list(args_dict)]): | ||
with pytest.raises(SystemExit, match="0"): | ||
cli_main() | ||
|
||
# directories for compdef model should exist | ||
tmp_dir = pathlib.Path(tmp_init_dir) | ||
model_dirs = [d.name for d in tmp_dir.iterdir() if not is_hidden(d)] | ||
expected = InitEntrypoint.MODEL_DIRS[args_dict["oscal-model"]] | ||
assert sorted(model_dirs) == sorted(expected) | ||
|
||
# directories for github workflows should exist | ||
workflow_dir = tmp_dir.joinpath(".github/workflows") | ||
workflow_files = [f.name for f in workflow_dir.iterdir()] | ||
expected = InitEntrypoint.PROVIDER_TEMPLATES[args_dict["provider"]][ | ||
args_dict["oscal-model"] | ||
] | ||
assert sorted(workflow_files) == sorted(expected) | ||
|
||
assert any( | ||
record.levelno == logging.INFO | ||
and f"Initialized trestlebot project successfully in {tmp_init_dir}" | ||
in record.message | ||
for record in caplog.records | ||
) | ||
|
||
|
||
def test_call_trestle_init(tmp_init_dir: str) -> None: | ||
"""Tests for expected results of calling trestle init""" | ||
parser = argparse.ArgumentParser() | ||
args = argparse.Namespace(verbose=0, working_dir=tmp_init_dir) | ||
InitEntrypoint(parser=parser)._call_trestle_init(args) | ||
tmp_dir = pathlib.Path(tmp_init_dir) | ||
trestle_dir = tmp_dir / pathlib.Path(TRESTLE_CONFIG_DIR) | ||
keep_file = trestle_dir / pathlib.Path(TRESTLE_KEEP_FILE) | ||
assert keep_file.exists() is True |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.