Skip to content

Commit a9a900a

Browse files
authored
Refactor aws_ssm display and verbosity-related methods (#2264)
SUMMARY Resolves ACA-2239 Methods like self._vvvv, self._vvv, etc. have been refactored into a single method, including code from the _display() method. ISSUE TYPE Feature Pull Request COMPONENT NAME connection/aws_ssm Reviewed-by: Bianca Henderson <[email protected]> Reviewed-by: Bikouo Aubin Reviewed-by: Mike Graves <[email protected]> Reviewed-by: Rahmanim Benny <[email protected]>
1 parent d9899d0 commit a9a900a

File tree

5 files changed

+103
-54
lines changed

5 files changed

+103
-54
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
minor_changes:
2+
- aws_ssm - Refactor display/verbosity-related methods in aws_ssm to simplify the code and avoid repetition (https://github.com/ansible-collections/community.aws/pull/2264).

plugins/connection/aws_ssm.py

+42-40
Original file line numberDiff line numberDiff line change
@@ -383,7 +383,7 @@ def wrapped(self, *args: Any, **kwargs: Any) -> Any:
383383
for attempt in range(remaining_tries):
384384
try:
385385
return_tuple = func(self, *args, **kwargs)
386-
self._vvvv(f"ssm_retry: (success) {to_text(return_tuple)}")
386+
self.verbosity_display(4, f"ssm_retry: (success) {to_text(return_tuple)}")
387387
break
388388

389389
except (AnsibleConnectionFailure, Exception) as e:
@@ -400,7 +400,7 @@ def wrapped(self, *args: Any, **kwargs: Any) -> Any:
400400
f"from cmd ({cmd_summary}),pausing for {pause} seconds"
401401
)
402402

403-
self._vv(msg)
403+
self.verbosity_display(2, msg)
404404

405405
time.sleep(pause)
406406

@@ -511,7 +511,8 @@ def _init_clients(self) -> None:
511511
Initializes required AWS clients (SSM and S3).
512512
Delegates client initialization to specialized methods.
513513
"""
514-
self._vvvv("INITIALIZE BOTO3 CLIENTS")
514+
515+
self.verbosity_display(4, "INITIALIZE BOTO3 CLIENTS")
515516
profile_name = self.get_option("profile") or ""
516517
region_name = self.get_option("region")
517518

@@ -520,7 +521,7 @@ def _init_clients(self) -> None:
520521

521522
# Initialize S3 client
522523
s3_endpoint_url, s3_region_name = self.s3_manager.get_bucket_endpoint()
523-
self._vvvv(f"SETUP BOTO3 CLIENTS: S3 {s3_endpoint_url}")
524+
self.verbosity_display(4, f"SETUP BOTO3 CLIENTS: S3 {s3_endpoint_url}")
524525
self.s3_manager.initialize_client(
525526
region_name=s3_region_name, endpoint_url=s3_endpoint_url, profile_name=profile_name
526527
)
@@ -540,35 +541,36 @@ def _initialize_ssm_client(self, region_name: Optional[str], profile_name: str)
540541
None
541542
"""
542543

543-
self._vvvv("SETUP BOTO3 CLIENTS: SSM")
544+
self.verbosity_display(4, "SETUP BOTO3 CLIENTS: SSM")
544545
self._client = self._get_boto_client(
545546
"ssm",
546547
region_name=region_name,
547548
profile_name=profile_name,
548549
)
549550

550-
def _display(self, f: Any, message: str) -> None:
551+
def verbosity_display(self, level: int, message: str) -> None:
552+
"""
553+
Displays the given message depending on the verbosity level.
554+
555+
:param message: The message to display.
556+
:param display_level: The verbosity level (1-4).
557+
558+
:return: None
559+
"""
551560
if self.host:
552561
host_args = {"host": self.host}
553562
else:
554563
host_args = {}
555-
f(to_text(message), **host_args)
556-
557-
def _v(self, message: str) -> None:
558-
self._display(display.v, message)
559-
560-
def _vv(self, message: str) -> None:
561-
self._display(display.vv, message)
562564

563-
def _vvv(self, message: str) -> None:
564-
self._display(display.vvv, message)
565+
verbosity_level = {1: display.v, 2: display.vv, 3: display.vvv, 4: display.vvvv}
565566

566-
def _vvvv(self, message: str) -> None:
567-
self._display(display.vvvv, message)
567+
if level not in verbosity_level.keys():
568+
raise AnsibleError(f"Invalid verbosity level: {level}")
569+
verbosity_level[level](to_text(message), **host_args)
568570

569571
def reset(self) -> Any:
570572
"""start a fresh ssm session"""
571-
self._vvvv("reset called on ssm connection")
573+
self.verbosity_display(4, "reset called on ssm connection")
572574
self.close()
573575
return self.start_session()
574576

@@ -601,13 +603,13 @@ def get_executable(self) -> str:
601603
def start_session(self):
602604
"""start ssm session"""
603605

604-
self._vvv(f"ESTABLISH SSM CONNECTION TO: {self.instance_id}")
606+
self.verbosity_display(3, f"ESTABLISH SSM CONNECTION TO: {self.instance_id}")
605607

606608
executable = self.get_executable()
607609

608610
self._init_clients()
609611

610-
self._vvvv(f"START SSM SESSION: {self.instance_id}")
612+
self.verbosity_display(4, f"START SSM SESSION: {self.instance_id}")
611613
start_session_args = dict(Target=self.instance_id, Parameters={})
612614
document_name = self.get_option("ssm_document")
613615
if document_name is not None:
@@ -627,7 +629,7 @@ def start_session(self):
627629
self._client.meta.endpoint_url,
628630
]
629631

630-
self._vvvv(f"SSM COMMAND: {to_text(cmd)}")
632+
self.verbosity_display(4, f"SSM COMMAND: {to_text(cmd)}")
631633

632634
stdout_r, stdout_w = pty.openpty()
633635
self._session = subprocess.Popen(
@@ -645,7 +647,7 @@ def start_session(self):
645647
# For non-windows Hosts: Ensure the session has started, and disable command echo and prompt.
646648
self._prepare_terminal()
647649

648-
self._vvvv(f"SSM CONNECTION ID: {self._session_id}") # pylint: disable=unreachable
650+
self.verbosity_display(4, f"SSM CONNECTION ID: {self._session_id}") # pylint: disable=unreachable
649651

650652
return self._session
651653

@@ -671,7 +673,7 @@ def poll(self, label: str, cmd: str) -> NoReturn:
671673
timeout = self.get_option("ssm_timeout")
672674
while self._session.poll() is None:
673675
remaining = start + timeout - round(time.time())
674-
self._vvvv(f"{label} remaining: {remaining} second(s)")
676+
self.verbosity_display(4, f"{label} remaining: {remaining} second(s)")
675677
if remaining < 0:
676678
self._has_timeout = True
677679
raise AnsibleConnectionFailure(f"{label} command '{cmd}' timeout on host: {self.instance_id}")
@@ -697,7 +699,7 @@ def exec_communicate(self, cmd: str, mark_start: str, mark_begin: str, mark_end:
697699
continue
698700

699701
line = filter_ansi(self._stdout.readline(), self.is_windows)
700-
self._vvvv(f"EXEC stdout line: \n{line}")
702+
self.verbosity_display(4, f"EXEC stdout line: \n{line}")
701703

702704
if not begin and self.is_windows:
703705
win_line = win_line + line
@@ -710,9 +712,9 @@ def exec_communicate(self, cmd: str, mark_start: str, mark_begin: str, mark_end:
710712
continue
711713
if begin:
712714
if mark_end in line:
713-
self._vvvv(f"POST_PROCESS: \n{to_text(stdout)}")
715+
self.verbosity_display(4, f"POST_PROCESS: \n{to_text(stdout)}")
714716
returncode, stdout = self._post_process(stdout, mark_begin)
715-
self._vvvv(f"POST_PROCESSED: \n{to_text(stdout)}")
717+
self.verbosity_display(4, f"POST_PROCESSED: \n{to_text(stdout)}")
716718
break
717719
stdout = stdout + line
718720

@@ -731,7 +733,7 @@ def exec_command(self, cmd: str, in_data: bool = None, sudoable: bool = True) ->
731733

732734
super().exec_command(cmd, in_data=in_data, sudoable=sudoable)
733735

734-
self._vvv(f"EXEC: {to_text(cmd)}")
736+
self.verbosity_display(3, f"EXEC: {to_text(cmd)}")
735737

736738
mark_begin = self.generate_mark()
737739
if self.is_windows:
@@ -758,10 +760,10 @@ def _ensure_ssm_session_has_started(self) -> None:
758760
for poll_result in self.poll("START SSM SESSION", "start_session"):
759761
if poll_result:
760762
stdout += to_text(self._stdout.read(1024))
761-
self._vvvv(f"START SSM SESSION stdout line: \n{to_bytes(stdout)}")
763+
self.verbosity_display(4, f"START SSM SESSION stdout line: \n{to_bytes(stdout)}")
762764
match = str(stdout).find("Starting session with SessionId")
763765
if match != -1:
764-
self._vvvv("START SSM SESSION startup output received")
766+
self.verbosity_display(4, "START SSM SESSION startup output received")
765767
break
766768

767769
def _disable_prompt_command(self) -> None:
@@ -774,14 +776,14 @@ def _disable_prompt_command(self) -> None:
774776
disable_prompt_reply = re.compile(r"\r\r\n" + re.escape(end_mark) + r"\r\r\n", re.MULTILINE)
775777

776778
# Send command
777-
self._vvvv(f"DISABLE PROMPT Disabling Prompt: \n{disable_prompt_cmd}")
779+
self.verbosity_display(4, f"DISABLE PROMPT Disabling Prompt: \n{disable_prompt_cmd}")
778780
self._session.stdin.write(disable_prompt_cmd)
779781

780782
stdout = ""
781783
for poll_result in self.poll("DISABLE PROMPT", disable_prompt_cmd):
782784
if poll_result:
783785
stdout += to_text(self._stdout.read(1024))
784-
self._vvvv(f"DISABLE PROMPT stdout line: \n{to_bytes(stdout)}")
786+
self.verbosity_display(4, f"DISABLE PROMPT stdout line: \n{to_bytes(stdout)}")
785787
if disable_prompt_reply.search(stdout):
786788
break
787789

@@ -790,14 +792,14 @@ def _disable_echo_command(self) -> None:
790792
disable_echo_cmd = to_bytes("stty -echo\n", errors="surrogate_or_strict")
791793

792794
# Send command
793-
self._vvvv(f"DISABLE ECHO Disabling Prompt: \n{disable_echo_cmd}")
795+
self.verbosity_display(4, f"DISABLE ECHO Disabling Prompt: \n{disable_echo_cmd}")
794796
self._session.stdin.write(disable_echo_cmd)
795797

796798
stdout = ""
797799
for poll_result in self.poll("DISABLE ECHO", disable_echo_cmd):
798800
if poll_result:
799801
stdout += to_text(self._stdout.read(1024))
800-
self._vvvv(f"DISABLE ECHO stdout line: \n{to_bytes(stdout)}")
802+
self.verbosity_display(4, f"DISABLE ECHO stdout line: \n{to_bytes(stdout)}")
801803
match = str(stdout).find("stty -echo")
802804
if match != -1:
803805
break
@@ -817,7 +819,7 @@ def _prepare_terminal(self) -> None:
817819
# Disable prompt command
818820
self._disable_prompt_command() # pylint: disable=unreachable
819821

820-
self._vvvv("PRE Terminal configured") # pylint: disable=unreachable
822+
self.verbosity_display(4, "PRE Terminal configured") # pylint: disable=unreachable
821823

822824
def _wrap_command(self, cmd: str, mark_start: str, mark_end: str) -> str:
823825
"""Wrap command so stdout and status can be extracted"""
@@ -833,7 +835,7 @@ def _wrap_command(self, cmd: str, mark_start: str, mark_end: str) -> str:
833835
f"printf '\\n%s\\n%s\\n' \"$?\" '{mark_end}';\n"
834836
) # fmt: skip
835837

836-
self._vvvv(f"_wrap_command: \n'{to_text(cmd)}'")
838+
self.verbosity_display(4, f"_wrap_command: \n'{to_text(cmd)}'")
837839
return cmd
838840

839841
def _post_process(self, stdout: str, mark_begin: str) -> Tuple[str, str]:
@@ -881,7 +883,7 @@ def _flush_stderr(self, session_process) -> str:
881883
if not poll_stderr.poll(1):
882884
break
883885
line = session_process.stderr.readline()
884-
self._vvvv(f"stderr line: {to_text(line)}")
886+
self.verbosity_display(4, f"stderr line: {to_text(line)}")
885887
stderr = stderr + line
886888

887889
return stderr
@@ -1084,7 +1086,7 @@ def put_file(self, in_path: str, out_path: str) -> Tuple[int, str, str]:
10841086

10851087
super().put_file(in_path, out_path)
10861088

1087-
self._vvv(f"PUT {in_path} TO {out_path}")
1089+
self.verbosity_display(3, f"PUT {in_path} TO {out_path}")
10881090
if not os.path.exists(to_bytes(in_path, errors="surrogate_or_strict")):
10891091
raise AnsibleFileNotFound(f"file or module does not exist: {in_path}")
10901092

@@ -1095,19 +1097,19 @@ def fetch_file(self, in_path: str, out_path: str) -> Tuple[int, str, str]:
10951097

10961098
super().fetch_file(in_path, out_path)
10971099

1098-
self._vvv(f"FETCH {in_path} TO {out_path}")
1100+
self.verbosity_display(3, f"FETCH {in_path} TO {out_path}")
10991101
return self._file_transport_command(in_path, out_path, "get")
11001102

11011103
def close(self) -> None:
11021104
"""terminate the connection"""
11031105
if self._session_id:
1104-
self._vvv(f"CLOSING SSM CONNECTION TO: {self.instance_id}")
1106+
self.verbosity_display(3, f"CLOSING SSM CONNECTION TO: {self.instance_id}")
11051107
if self._has_timeout:
11061108
self._session.terminate()
11071109
else:
11081110
cmd = b"\nexit\n"
11091111
self._session.communicate(cmd)
11101112

1111-
self._vvvv(f"TERMINATE SSM SESSION: {self._session_id}")
1113+
self.verbosity_display(4, f"TERMINATE SSM SESSION: {self._session_id}")
11121114
self._client.terminate_session(SessionId=self._session_id)
11131115
self._session_id = ""

plugins/plugin_utils/s3clientmanager.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -38,7 +38,7 @@ def get_bucket_endpoint(self) -> Tuple[str, str]:
3838
"""
3939
region_name = self.connection.get_option("region") or "us-east-1"
4040
profile_name = self.connection.get_option("profile") or ""
41-
self.connection._vvvv("_get_bucket_endpoint: S3 (global)")
41+
self.connection.verbosity_display(4, "_get_bucket_endpoint: S3 (global)")
4242
tmp_s3_client = self.get_s3_client(
4343
region_name=region_name,
4444
profile_name=profile_name,
@@ -56,7 +56,7 @@ def get_bucket_endpoint(self) -> Tuple[str, str]:
5656
return self.connection.get_option("bucket_endpoint_url"), bucket_region
5757

5858
# Create another client for the region the bucket lives in, so we can nab the endpoint URL
59-
self.connection._vvvv(f"_get_bucket_endpoint: S3 (bucket region) - {bucket_region}")
59+
self.connection.verbosity_display(4, f"_get_bucket_endpoint: S3 (bucket region) - {bucket_region}")
6060
s3_bucket_client = self.get_s3_client(
6161
region_name=bucket_region,
6262
profile_name=profile_name,

tests/unit/plugins/connection/aws_ssm/conftest.py

+2-11
Original file line numberDiff line numberDiff line change
@@ -40,17 +40,8 @@ def connection_init(*args, **kwargs):
4040
def display_msg(msg):
4141
print("--- AWS SSM CONNECTION --- ", msg)
4242

43-
connection._v = MagicMock()
44-
connection._v.side_effect = display_msg
45-
46-
connection._vv = MagicMock()
47-
connection._vv.side_effect = display_msg
48-
49-
connection._vvv = MagicMock()
50-
connection._vvv.side_effect = display_msg
51-
52-
connection._vvvv = MagicMock()
53-
connection._vvvv.side_effect = display_msg
43+
connection.verbosity_display = MagicMock()
44+
connection.verbosity_display.side_effect = lambda level, msg: display_msg(msg)
5445

5546
connection.get_option = MagicMock()
5647
return connection

tests/unit/plugins/connection/aws_ssm/test_aws_ssm.py

+55-1
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@
66

77
import pytest
88

9+
from ansible.errors import AnsibleError
910
from ansible.playbook.play_context import PlayContext
1011
from ansible.plugins.loader import connection_loader
1112

@@ -258,7 +259,7 @@ def test_generate_mark(self):
258259

259260
@pytest.mark.parametrize("is_windows", [False, True])
260261
def test_generate_commands(self, is_windows):
261-
"""Testing command generation on Windows systems"""
262+
"""Testing command generation on both Windows and non-Windows systems"""
262263
pc = PlayContext()
263264
new_stdin = StringIO()
264265
conn = connection_loader.get("community.aws.aws_ssm", pc, new_stdin)
@@ -307,6 +308,59 @@ def test_generate_commands(self, is_windows):
307308
assert isinstance(test_command_generation[0], list)
308309
assert isinstance(test_command_generation[0][0], dict)
309310

311+
@pytest.mark.parametrize(
312+
"message,level,method",
313+
[
314+
("test message 1", 1, "v"),
315+
("test message 2", 2, "vv"),
316+
("test message 3", 3, "vvv"),
317+
("test message 4", 4, "vvvv"),
318+
],
319+
)
320+
def test_verbosity_diplay(self, message, level, method):
321+
"""Testing verbosity levels"""
322+
play_context = MagicMock()
323+
play_context.shell = "sh"
324+
conn = Connection(play_context)
325+
conn.host = "test-host" # Test with host set
326+
327+
with patch("ansible_collections.community.aws.plugins.connection.aws_ssm.display") as mock_display:
328+
conn.verbosity_display(level, message)
329+
# Verify the correct display method was called with expected args
330+
mock_method = getattr(mock_display, method)
331+
mock_method.assert_called_once_with(message, host="test-host")
332+
333+
# Test without host set
334+
conn.host = None
335+
conn.verbosity_display(1, "no host message")
336+
mock_display.v.assert_called_with("no host message")
337+
338+
# Test exception is raised when verbosity level is not an accepted value
339+
with pytest.raises(AnsibleError):
340+
conn.verbosity_display("invalid value", "test message")
341+
342+
def test_poll_verbosity(self):
343+
"""Test poll method verbosity display"""
344+
pc = PlayContext()
345+
new_stdin = StringIO()
346+
conn = connection_loader.get("community.aws.aws_ssm", pc, new_stdin)
347+
348+
conn._session = MagicMock()
349+
conn._session.poll.return_value = None
350+
conn.get_option = MagicMock(return_value=10) # ssm_timeout
351+
conn.poll_stdout = MagicMock()
352+
conn.instance_id = "i-1234567890"
353+
conn.host = conn.instance_id
354+
355+
with patch("time.time", return_value=100), patch.object(conn, "verbosity_display") as mock_display:
356+
poll_gen = conn.poll("TEST", "test command")
357+
# Advance generator twice to trigger the verbosity message
358+
next(poll_gen)
359+
next(poll_gen)
360+
361+
# Verify verbosity message contains remaining time
362+
mock_display.assert_called_with(4, "TEST remaining: 10 second(s)")
363+
310364

311365
class TestS3ClientManager:
312366
"""

0 commit comments

Comments
 (0)