From 9f7d5d9ab401b623a4d846f3a1516caf6b1e20cf Mon Sep 17 00:00:00 2001 From: Garfield Lee Freeman Date: Wed, 8 Nov 2023 09:19:13 +0100 Subject: [PATCH 1/3] feat(PanDevice): add `is_ready()` --- panos/base.py | 31 +++++++++++++++++++++++++++++++ tests/test_base.py | 31 +++++++++++++++++++++++++++++++ 2 files changed, 62 insertions(+) diff --git a/panos/base.py b/panos/base.py index 1a9cc90c..03a81fce 100644 --- a/panos/base.py +++ b/panos/base.py @@ -5589,3 +5589,34 @@ def whoami(self): if name is not None and is_self: return name + + def is_ready(self, minutes=None, seconds=None): + """Runs "show chassis-ready" until the PAN-OS management plane is up. + + Args: + minutes (int): The number of minutes to wait before giving up. + seconds (int): The number of seconds to wait before giving up. + + Returns: + True if PAN-OS is ready, or False if a timeout was reached. + """ + end = None + if minutes is not None or seconds is not None: + end = datetime.datetime.now() + datetime.timedelta(minutes=minutes or 0, seconds=seconds or 0) + + while True: + response = None + try: + response = self.op("show chassis-ready") + except (err.PanURLError, pan.xapi.PanXapiError, err.PanDeviceXapiError): + pass + else: + ready_status = response.find(".//result").text.strip() + if ready_status.lower() == "yes": + return True + + if end is not None and datetime.datetime.now() >= end: + return False + + # Device isn't up yet, retry after sleeping. + time.sleep(2) diff --git a/tests/test_base.py b/tests/test_base.py index a432ccc2..422eb0a4 100644 --- a/tests/test_base.py +++ b/tests/test_base.py @@ -1620,5 +1620,36 @@ def test_delete_extreme_members(self): self.assertTrue(dev.xapi.delete.call_count > 2) +class TestIsReady(unittest.TestCase): + @mock.patch("time.sleep") + def test_ok(self, mocksleep): + fw = Base.PanDevice("127.0.0.1", "admin", "secret", api_key="apikey") + fw.xapi.op = mock.Mock(side_effect=[ + Err.PanURLError, + pan.xapi.PanXapiError, + Err.PanXapiError, + ET.fromstring("yes"), + ValueError, + ]) + + ans = fw.is_ready() + + assert ans == True + assert mocksleep.call_count == 3 + + @mock.patch("time.sleep") + def test_times_out(self, mocksleep): + fw = Base.PanDevice("127.0.0.1", "admin", "secret", api_key="apikey") + fw.xapi.op = mock.Mock(side_effect=[ + Err.PanURLError, + ValueError, + ]) + + ans = fw.is_ready(seconds=0) + + assert ans == False + assert mocksleep.call_count == 0 + + if __name__ == "__main__": unittest.main() From 0ee5dcc6cac83c258a17a257f2d00e407fdee6f8 Mon Sep 17 00:00:00 2001 From: Garfield Lee Freeman Date: Wed, 8 Nov 2023 09:36:23 +0100 Subject: [PATCH 2/3] trying to fix the black issues --- Makefile | 2 +- tests/test_base.py | 26 +++++++++++++++----------- 2 files changed, 16 insertions(+), 12 deletions(-) diff --git a/Makefile b/Makefile index e6aed882..72a264b1 100644 --- a/Makefile +++ b/Makefile @@ -53,7 +53,7 @@ format: check-format: isort --recursive --atomic --check-only panos - black --check . + black --check --diff . test: pytest diff --git a/tests/test_base.py b/tests/test_base.py index 422eb0a4..9429d7d1 100644 --- a/tests/test_base.py +++ b/tests/test_base.py @@ -1624,13 +1624,15 @@ class TestIsReady(unittest.TestCase): @mock.patch("time.sleep") def test_ok(self, mocksleep): fw = Base.PanDevice("127.0.0.1", "admin", "secret", api_key="apikey") - fw.xapi.op = mock.Mock(side_effect=[ - Err.PanURLError, - pan.xapi.PanXapiError, - Err.PanXapiError, - ET.fromstring("yes"), - ValueError, - ]) + fw.xapi.op = mock.Mock( + side_effect=[ + Err.PanURLError, + pan.xapi.PanXapiError, + Err.PanXapiError, + ET.fromstring("yes"), + ValueError, + ], + ) ans = fw.is_ready() @@ -1640,10 +1642,12 @@ def test_ok(self, mocksleep): @mock.patch("time.sleep") def test_times_out(self, mocksleep): fw = Base.PanDevice("127.0.0.1", "admin", "secret", api_key="apikey") - fw.xapi.op = mock.Mock(side_effect=[ - Err.PanURLError, - ValueError, - ]) + fw.xapi.op = mock.Mock( + side_effect=[ + Err.PanURLError, + ValueError, + ], + ) ans = fw.is_ready(seconds=0) From b73efa69257e1659d6df315453404b78c0cc1ebc Mon Sep 17 00:00:00 2001 From: Garfield Lee Freeman Date: Wed, 8 Nov 2023 09:40:02 +0100 Subject: [PATCH 3/3] fixing black issues finally --- panos/base.py | 4 +++- tests/test_base.py | 7 +------ 2 files changed, 4 insertions(+), 7 deletions(-) diff --git a/panos/base.py b/panos/base.py index 03a81fce..120b91a8 100644 --- a/panos/base.py +++ b/panos/base.py @@ -5602,7 +5602,9 @@ def is_ready(self, minutes=None, seconds=None): """ end = None if minutes is not None or seconds is not None: - end = datetime.datetime.now() + datetime.timedelta(minutes=minutes or 0, seconds=seconds or 0) + end = datetime.datetime.now() + datetime.timedelta( + minutes=minutes or 0, seconds=seconds or 0, + ) while True: response = None diff --git a/tests/test_base.py b/tests/test_base.py index 9429d7d1..235a55d4 100644 --- a/tests/test_base.py +++ b/tests/test_base.py @@ -1642,12 +1642,7 @@ def test_ok(self, mocksleep): @mock.patch("time.sleep") def test_times_out(self, mocksleep): fw = Base.PanDevice("127.0.0.1", "admin", "secret", api_key="apikey") - fw.xapi.op = mock.Mock( - side_effect=[ - Err.PanURLError, - ValueError, - ], - ) + fw.xapi.op = mock.Mock(side_effect=[Err.PanURLError, ValueError,],) ans = fw.is_ready(seconds=0)