dda-python-terraform/test/test_terraform.py

472 lines
15 KiB
Python
Raw Normal View History

2020-10-19 22:30:02 +00:00
import fnmatch
import logging
2020-10-19 22:30:02 +00:00
import os
import re
2016-12-20 16:55:55 +00:00
import shutil
2020-10-19 22:30:02 +00:00
from contextlib import contextmanager
2020-10-20 22:08:58 +00:00
from io import StringIO
2021-01-21 10:07:00 +00:00
from typing import Callable
2020-10-19 22:30:02 +00:00
import pytest
2021-01-21 10:07:00 +00:00
2020-10-20 22:08:58 +00:00
from python_terraform import IsFlagged, IsNotFlagged, Terraform, TerraformCommandError
logging.basicConfig(level=logging.DEBUG)
2017-05-09 09:47:36 +00:00
root_logger = logging.getLogger()
current_path = os.path.dirname(os.path.realpath(__file__))
FILE_PATH_WITH_SPACE_AND_SPACIAL_CHARS = "test 'test.out!"
STRING_CASES = [
2020-10-19 22:30:02 +00:00
[
lambda x: x.generate_cmd_string("apply", "the_folder", no_color=IsFlagged),
"terraform apply -no-color the_folder",
],
[
lambda x: x.generate_cmd_string(
"push", "path", vcs=True, token="token", atlas_address="url"
),
"terraform push -vcs=true -token=token -atlas-address=url path",
],
]
CMD_CASES = [
2020-10-19 22:30:02 +00:00
[
"method",
"expected_output",
"expected_ret_code",
"expected_exception",
"expected_logs",
"folder",
],
[
[
2020-10-19 22:30:02 +00:00
lambda x: x.cmd(
"plan", "var_to_output", no_color=IsFlagged, var={"test_var": "test"}
),
2017-10-16 16:03:45 +00:00
# Expected output varies by terraform version
2020-10-20 22:08:58 +00:00
"Plan: 0 to add, 0 to change, 0 to destroy.",
2017-05-09 09:47:36 +00:00
0,
False,
2020-10-19 22:30:02 +00:00
"",
"var_to_output",
],
# try import aws instance
[
2020-10-19 22:30:02 +00:00
lambda x: x.cmd(
"import", "aws_instance.foo", "i-abcd1234", no_color=IsFlagged
),
"",
2017-05-09 09:47:36 +00:00
1,
False,
2020-10-19 22:30:02 +00:00
"Command: terraform import -no-color aws_instance.foo i-abcd1234",
"",
],
# try import aws instance with raise_on_error
[
2020-10-19 22:30:02 +00:00
lambda x: x.cmd(
"import",
"aws_instance.foo",
"i-abcd1234",
no_color=IsFlagged,
raise_on_error=True,
),
"",
1,
True,
2020-10-19 22:30:02 +00:00
"Command: terraform import -no-color aws_instance.foo i-abcd1234",
"",
],
# test with space and special character in file path
[
2020-10-19 22:30:02 +00:00
lambda x: x.cmd(
"plan", "var_to_output", out=FILE_PATH_WITH_SPACE_AND_SPACIAL_CHARS
),
"",
0,
False,
2020-10-19 22:30:02 +00:00
"",
"var_to_output",
],
# test workspace command (commands with subcommand)
[
2020-10-19 22:30:02 +00:00
lambda x: x.cmd("workspace", "show", no_color=IsFlagged),
"",
0,
False,
2020-10-19 22:30:02 +00:00
"Command: terraform workspace show -no-color",
"",
],
2020-10-19 22:30:02 +00:00
],
]
2015-12-31 07:15:51 +00:00
2020-10-19 22:30:02 +00:00
@pytest.fixture(scope="function")
2016-12-20 16:55:55 +00:00
def fmt_test_file(request):
2020-10-19 22:30:02 +00:00
target = os.path.join(current_path, "bad_fmt", "test.backup")
orgin = os.path.join(current_path, "bad_fmt", "test.tf")
shutil.copy(orgin, target)
2016-12-20 16:55:55 +00:00
def td():
shutil.move(target, orgin)
request.addfinalizer(td)
return
2015-12-31 07:15:51 +00:00
2017-05-09 09:47:36 +00:00
@pytest.fixture()
2021-01-21 10:07:00 +00:00
def string_logger(request) -> Callable[..., str]:
2017-05-09 09:47:36 +00:00
log_stream = StringIO()
handler = logging.StreamHandler(log_stream)
root_logger.addHandler(handler)
def td():
root_logger.removeHandler(handler)
2017-08-04 21:14:57 +00:00
log_stream.close()
2017-05-09 09:47:36 +00:00
request.addfinalizer(td)
2017-08-04 21:14:57 +00:00
return lambda: str(log_stream.getvalue())
2017-05-09 09:47:36 +00:00
@pytest.fixture()
def workspace_setup_teardown():
"""
Fixture used in workspace related tests
Create and tear down a workspace
*Use as a contextmanager*
"""
2020-10-19 22:30:02 +00:00
@contextmanager
def wrapper(workspace_name, create=True, delete=True, *args, **kwargs):
tf = Terraform(working_dir=current_path)
tf.init()
if create:
tf.create_workspace(workspace_name, *args, **kwargs)
yield tf
if delete:
2020-10-19 22:30:02 +00:00
tf.set_workspace("default")
tf.delete_workspace(workspace_name)
yield wrapper
class TestTerraform(object):
2021-01-21 10:07:00 +00:00
def teardown_method(self, _) -> None:
""" teardown any state that was previously setup with a setup_method
call.
"""
2020-10-19 22:30:02 +00:00
exclude = ["test_tfstate_file", "test_tfstate_file2", "test_tfstate_file3"]
2021-01-21 10:07:00 +00:00
def purge(dir: str, pattern: str) -> None:
2017-08-04 23:02:49 +00:00
for root, dirnames, filenames in os.walk(dir):
dirnames[:] = [d for d in dirnames if d not in exclude]
2017-08-04 23:02:49 +00:00
for filename in fnmatch.filter(filenames, pattern):
f = os.path.join(root, filename)
os.remove(f)
for dirname in fnmatch.filter(dirnames, pattern):
d = os.path.join(root, dirname)
shutil.rmtree(d)
2020-10-19 22:30:02 +00:00
purge(".", "*.tfstate")
purge(".", "*.tfstate.backup")
purge(".", "*.terraform")
purge(".", FILE_PATH_WITH_SPACE_AND_SPACIAL_CHARS)
2020-10-19 22:30:02 +00:00
@pytest.mark.parametrize(["method", "expected"], STRING_CASES)
2021-01-21 10:07:00 +00:00
def test_generate_cmd_string(self, method: Callable[..., str], expected: str):
2016-12-20 11:58:04 +00:00
tf = Terraform(working_dir=current_path)
result = method(tf)
strs = expected.split()
for s in strs:
assert s in result
@pytest.mark.parametrize(*CMD_CASES)
2020-10-19 22:30:02 +00:00
def test_cmd(
self,
2021-01-21 10:07:00 +00:00
method: Callable[..., str],
expected_output: str,
expected_ret_code: int,
expected_exception: bool,
expected_logs: str,
string_logger: Callable[..., str],
2020-10-19 22:30:02 +00:00
folder,
):
tf = Terraform(working_dir=current_path)
2017-08-04 23:17:52 +00:00
tf.init(folder)
try:
2021-01-21 10:07:00 +00:00
ret, out, _ = method(tf)
2017-10-16 16:03:45 +00:00
assert not expected_exception
except TerraformCommandError as e:
2017-10-16 16:03:45 +00:00
assert expected_exception
ret = e.returncode
out = e.out
2020-10-19 22:24:33 +00:00
2017-08-04 21:14:57 +00:00
logs = string_logger()
2020-10-19 22:30:02 +00:00
logs = logs.replace("\n", "")
2021-01-21 10:07:00 +00:00
assert expected_output in out
assert expected_ret_code == ret
2017-05-09 09:47:36 +00:00
assert expected_logs in logs
2015-12-31 07:15:51 +00:00
@pytest.mark.parametrize(
("folder", "variables", "var_files", "expected_output", "options"),
[
2020-10-19 22:30:02 +00:00
("var_to_output", {"test_var": "test"}, None, "test_output=test", {}),
(
"var_to_output",
{"test_list_var": ["c", "d"]},
None,
2020-10-20 22:08:58 +00:00
'test_list_output=["c","d",]',
2020-10-19 22:30:02 +00:00
{},
),
(
"var_to_output",
{"test_map_var": {"c": "c", "d": "d"}},
None,
2020-10-20 22:08:58 +00:00
'test_map_output={"c"="c""d"="d"}',
2020-10-19 22:30:02 +00:00
{},
),
(
"var_to_output",
{"test_map_var": {"c": "c", "d": "d"}},
"var_to_output/test_map_var.json",
2020-10-20 22:08:58 +00:00
# Values are overriden
'test_map_output={"e"="e""f"="f"}',
2020-10-19 22:30:02 +00:00
{},
),
(
"var_to_output",
{},
None,
"\x1b[0m\x1b[1m\x1b[32mApplycomplete!",
{"no_color": IsNotFlagged},
),
],
)
def test_apply(self, folder, variables, var_files, expected_output, options):
2020-10-19 22:30:02 +00:00
tf = Terraform(
working_dir=current_path, variables=variables, var_file=var_files
)
2017-08-04 23:06:55 +00:00
tf.init(folder)
ret, out, err = tf.apply(folder, **options)
assert ret == 0
2020-10-19 22:30:02 +00:00
assert expected_output in out.replace("\n", "").replace(" ", "")
assert err == ""
def test_apply_with_var_file(self, string_logger):
tf = Terraform(working_dir=current_path)
tf.init()
2020-10-19 22:30:02 +00:00
tf.apply(var_file=os.path.join(current_path, "tfvar_file", "test.tfvars"))
logs = string_logger()
2020-10-19 22:30:02 +00:00
logs = logs.split("\n")
for log in logs:
2020-10-19 22:30:02 +00:00
if log.startswith("command: terraform apply"):
assert log.count("-var-file=") == 1
2016-12-20 16:43:44 +00:00
@pytest.mark.parametrize(
2020-10-19 22:30:02 +00:00
["cmd", "args", "options"],
2016-12-20 16:43:44 +00:00
[
# bool value
2020-10-19 22:30:02 +00:00
("fmt", ["bad_fmt"], {"list": False, "diff": False})
],
2016-12-20 16:43:44 +00:00
)
2016-12-20 16:55:55 +00:00
def test_options(self, cmd, args, options, fmt_test_file):
2016-12-20 16:43:44 +00:00
tf = Terraform(working_dir=current_path)
ret, out, err = getattr(tf, cmd)(*args, **options)
assert ret == 0
2020-10-19 22:30:02 +00:00
assert out == ""
def test_state_data(self):
2020-10-19 22:30:02 +00:00
cwd = os.path.join(current_path, "test_tfstate_file")
tf = Terraform(working_dir=cwd, state="tfstate.test")
tf.read_state_file()
2020-10-19 22:30:02 +00:00
assert tf.tfstate.modules[0]["path"] == ["root"]
2015-12-31 07:15:51 +00:00
def test_state_default(self):
2020-10-19 22:30:02 +00:00
cwd = os.path.join(current_path, "test_tfstate_file2")
tf = Terraform(working_dir=cwd)
tf.read_state_file()
2020-10-19 22:30:02 +00:00
assert tf.tfstate.modules[0]["path"] == ["default"]
def test_state_default_backend(self):
2020-10-19 22:30:02 +00:00
cwd = os.path.join(current_path, "test_tfstate_file3")
tf = Terraform(working_dir=cwd)
tf.read_state_file()
2020-10-19 22:30:02 +00:00
assert tf.tfstate.modules[0]["path"] == ["default_backend"]
def test_pre_load_state_data(self):
2020-10-19 22:30:02 +00:00
cwd = os.path.join(current_path, "test_tfstate_file")
tf = Terraform(working_dir=cwd, state="tfstate.test")
assert tf.tfstate.modules[0]["path"] == ["root"]
2015-12-31 07:15:51 +00:00
@pytest.mark.parametrize(
2020-10-19 22:30:02 +00:00
("folder", "variables"), [("var_to_output", {"test_var": "test"})]
)
def test_override_default(self, folder, variables):
2016-12-20 11:58:04 +00:00
tf = Terraform(working_dir=current_path, variables=variables)
2017-08-04 23:17:52 +00:00
tf.init(folder)
2020-10-19 22:30:02 +00:00
ret, out, err = tf.apply(
2020-10-20 22:08:58 +00:00
folder, var={"test_var": "test2"}, no_color=IsNotFlagged,
2020-10-19 22:30:02 +00:00
)
out = out.replace("\n", "")
assert "\x1b[0m\x1b[1m\x1b[32mApply" in out
out = tf.output("test_output")
assert "test2" in out
@pytest.mark.parametrize(("param"), [({}), ({"module": "test2"}),])
def test_output(self, param, string_logger):
2020-10-19 22:30:02 +00:00
tf = Terraform(working_dir=current_path, variables={"test_var": "test"})
tf.init("var_to_output")
tf.apply("var_to_output")
result = tf.output("test_output", **param)
regex = re.compile(
"terraform output (-module=test2 -json|-json -module=test2) test_output"
)
log_str = string_logger()
2017-08-04 21:14:57 +00:00
if param:
assert re.search(regex, log_str), log_str
2017-08-04 21:14:57 +00:00
else:
2020-10-19 22:30:02 +00:00
assert result == "test"
2020-10-19 22:30:02 +00:00
@pytest.mark.parametrize(("param"), [({}), ({"module": "test2"}),])
def test_output_all(self, param, string_logger):
2020-10-19 22:30:02 +00:00
tf = Terraform(working_dir=current_path, variables={"test_var": "test"})
tf.init("var_to_output")
tf.apply("var_to_output")
result = tf.output(**param)
regex = re.compile("terraform output (-module=test2 -json|-json -module=test2)")
log_str = string_logger()
if param:
assert re.search(regex, log_str), log_str
else:
2020-10-19 22:30:02 +00:00
assert result["test_output"]["value"] == "test"
def test_destroy(self):
2020-10-19 22:30:02 +00:00
tf = Terraform(working_dir=current_path, variables={"test_var": "test"})
tf.init("var_to_output")
ret, out, err = tf.destroy("var_to_output")
assert ret == 0
2020-10-19 22:30:02 +00:00
assert "Destroy complete! Resources: 0 destroyed." in out
@pytest.mark.parametrize(
2020-10-19 22:30:02 +00:00
("plan", "variables", "expected_ret"), [("vars_require_input", {}, 1)]
)
def test_plan(self, plan, variables, expected_ret):
tf = Terraform(working_dir=current_path, variables=variables)
ret, out, err = tf.plan(plan)
2017-01-04 15:46:15 +00:00
assert ret == expected_ret
def test_fmt(self, fmt_test_file):
2020-10-19 22:30:02 +00:00
tf = Terraform(working_dir=current_path, variables={"test_var": "test"})
ret, out, err = tf.fmt(diff=True)
assert ret == 0
2017-05-09 09:47:36 +00:00
def test_import(self, string_logger):
2017-05-09 09:51:52 +00:00
tf = Terraform(working_dir=current_path)
2020-10-19 22:30:02 +00:00
tf.import_cmd("aws_instance.foo", "i-abc1234", no_color=IsFlagged)
assert (
"Command: terraform import -no-color aws_instance.foo i-abc1234"
in string_logger()
)
2020-10-19 22:24:33 +00:00
def test_create_workspace(self, workspace_setup_teardown):
2020-10-19 22:30:02 +00:00
workspace_name = "test"
with workspace_setup_teardown(workspace_name, create=False) as tf:
2020-10-19 22:30:02 +00:00
ret, out, err = tf.create_workspace("test")
assert ret == 0
2020-10-19 22:30:02 +00:00
assert err == ""
2020-10-19 22:30:02 +00:00
def test_create_workspace_with_args(self, workspace_setup_teardown, string_logger):
workspace_name = "test"
state_file_path = os.path.join(
current_path, "test_tfstate_file2", "terraform.tfstate"
)
with workspace_setup_teardown(workspace_name, create=False) as tf:
2020-10-19 22:30:02 +00:00
ret, out, err = tf.create_workspace(
"test", current_path, no_color=IsFlagged
)
assert ret == 0
2020-10-19 22:30:02 +00:00
assert err == ""
logs = string_logger()
2020-10-19 22:30:02 +00:00
logs = logs.replace("\n", "")
expected_log = "Command: terraform workspace new -no-color test {}".format(
current_path
)
assert expected_log in logs
def test_set_workspace(self, workspace_setup_teardown):
2020-10-19 22:30:02 +00:00
workspace_name = "test"
with workspace_setup_teardown(workspace_name) as tf:
ret, out, err = tf.set_workspace(workspace_name)
assert ret == 0
2020-10-19 22:30:02 +00:00
assert err == ""
2019-05-20 18:34:42 +00:00
2020-10-19 22:30:02 +00:00
def test_set_workspace_with_args(self, workspace_setup_teardown, string_logger):
workspace_name = "test"
with workspace_setup_teardown(workspace_name) as tf:
2020-10-19 22:30:02 +00:00
ret, out, err = tf.set_workspace(
workspace_name, current_path, no_color=IsFlagged
)
assert ret == 0
2020-10-19 22:30:02 +00:00
assert err == ""
logs = string_logger()
2020-10-19 22:30:02 +00:00
logs = logs.replace("\n", "")
expected_log = "Command: terraform workspace select -no-color test {}".format(
current_path
)
assert expected_log in logs
def test_show_workspace(self, workspace_setup_teardown):
2020-10-19 22:30:02 +00:00
workspace_name = "test"
with workspace_setup_teardown(workspace_name) as tf:
ret, out, err = tf.show_workspace()
2019-05-20 18:34:42 +00:00
assert ret == 0
2020-10-19 22:30:02 +00:00
assert err == ""
def test_show_workspace_with_no_color(
2020-10-19 22:30:02 +00:00
self, workspace_setup_teardown, string_logger
):
2020-10-19 22:30:02 +00:00
workspace_name = "test"
with workspace_setup_teardown(workspace_name) as tf:
ret, out, err = tf.show_workspace(no_color=IsFlagged)
assert ret == 0
2020-10-19 22:30:02 +00:00
assert err == ""
logs = string_logger()
2020-10-19 22:30:02 +00:00
logs = logs.replace("\n", "")
expected_log = "Command: terraform workspace show -no-color"
assert expected_log in logs
def test_delete_workspace(self, workspace_setup_teardown):
2020-10-19 22:30:02 +00:00
workspace_name = "test"
with workspace_setup_teardown(workspace_name, delete=False) as tf:
2020-10-19 22:30:02 +00:00
tf.set_workspace("default")
ret, out, err = tf.delete_workspace(workspace_name)
assert ret == 0
2020-10-19 22:30:02 +00:00
assert err == ""
2020-10-19 22:30:02 +00:00
def test_delete_workspace_with_args(self, workspace_setup_teardown, string_logger):
workspace_name = "test"
with workspace_setup_teardown(workspace_name, delete=False) as tf:
2020-10-19 22:30:02 +00:00
tf.set_workspace("default")
ret, out, err = tf.delete_workspace(
2019-06-25 21:12:41 +00:00
workspace_name, current_path, force=IsFlagged,
)
assert ret == 0
2020-10-19 22:30:02 +00:00
assert err == ""
logs = string_logger()
2020-10-19 22:30:02 +00:00
logs = logs.replace("\n", "")
expected_log = "Command: terraform workspace delete -force test {}".format(
current_path
)
assert expected_log in logs