# -*- coding: utf-8 -*- # Copyright (c) 2021 Florian Dambrine <android.florian@gmail.com> # GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) # SPDX-License-Identifier: GPL-3.0-or-later from __future__ import absolute_import, division, print_function import sys from ansible.module_utils.common.dict_transformations import dict_merge from ansible.module_utils.six import iteritems from ansible_collections.community.general.plugins.modules import ( pritunl_org, ) from ansible_collections.community.general.tests.unit.compat.mock import patch from ansible_collections.community.general.tests.unit.plugins.module_utils.net_tools.pritunl.test_api import ( PritunlDeleteOrganizationMock, PritunlListOrganizationMock, PritunlListOrganizationAfterPostMock, PritunlPostOrganizationMock, ) from ansible_collections.community.general.tests.unit.plugins.modules.utils import ( AnsibleExitJson, AnsibleFailJson, ModuleTestCase, set_module_args, ) __metaclass__ = type class TestPritunlOrg(ModuleTestCase): def setUp(self): super(TestPritunlOrg, self).setUp() self.module = pritunl_org # Add backward compatibility if sys.version_info < (3, 2): self.assertRegex = self.assertRegexpMatches def tearDown(self): super(TestPritunlOrg, self).tearDown() def patch_add_pritunl_organization(self, **kwds): return patch( "ansible_collections.community.general.plugins.module_utils.net_tools.pritunl.api._post_pritunl_organization", autospec=True, **kwds ) def patch_delete_pritunl_organization(self, **kwds): return patch( "ansible_collections.community.general.plugins.module_utils.net_tools.pritunl.api._delete_pritunl_organization", autospec=True, **kwds ) def patch_get_pritunl_organizations(self, **kwds): return patch( "ansible_collections.community.general.plugins.module_utils.net_tools.pritunl.api._get_pritunl_organizations", autospec=True, **kwds ) def test_without_parameters(self): """Test without parameters""" set_module_args({}) with self.assertRaises(AnsibleFailJson): self.module.main() def test_present(self): """Test Pritunl organization creation.""" org_params = {"name": "NewOrg"} set_module_args( dict_merge( { "pritunl_api_token": "token", "pritunl_api_secret": "secret", "pritunl_url": "https://pritunl.domain.com", }, org_params, ) ) # Test creation with self.patch_get_pritunl_organizations( side_effect=PritunlListOrganizationMock ) as mock_get: with self.patch_add_pritunl_organization( side_effect=PritunlPostOrganizationMock ) as mock_add: with self.assertRaises(AnsibleExitJson) as create_result: self.module.main() create_exc = create_result.exception.args[0] self.assertTrue(create_exc["changed"]) self.assertEqual(create_exc["response"]["name"], org_params["name"]) self.assertEqual(create_exc["response"]["user_count"], 0) # Test module idempotency with self.patch_get_pritunl_organizations( side_effect=PritunlListOrganizationAfterPostMock ) as mock_get: with self.patch_add_pritunl_organization( side_effect=PritunlPostOrganizationMock ) as mock_add: with self.assertRaises(AnsibleExitJson) as idempotent_result: self.module.main() idempotent_exc = idempotent_result.exception.args[0] # Ensure both calls resulted in the same returned value # except for changed which should be false the second time for k, v in iteritems(idempotent_exc): if k == "changed": self.assertFalse(idempotent_exc[k]) else: self.assertEqual(create_exc[k], idempotent_exc[k]) def test_absent(self): """Test organization removal from Pritunl.""" org_params = {"name": "NewOrg"} set_module_args( dict_merge( { "state": "absent", "pritunl_api_token": "token", "pritunl_api_secret": "secret", "pritunl_url": "https://pritunl.domain.com", }, org_params, ) ) # Test deletion with self.patch_get_pritunl_organizations( side_effect=PritunlListOrganizationAfterPostMock ) as mock_get: with self.patch_delete_pritunl_organization( side_effect=PritunlDeleteOrganizationMock ) as mock_delete: with self.assertRaises(AnsibleExitJson) as delete_result: self.module.main() delete_exc = delete_result.exception.args[0] self.assertTrue(delete_exc["changed"]) self.assertEqual(delete_exc["response"], {}) # Test module idempotency with self.patch_get_pritunl_organizations( side_effect=PritunlListOrganizationMock ) as mock_get: with self.patch_delete_pritunl_organization( side_effect=PritunlDeleteOrganizationMock ) as mock_add: with self.assertRaises(AnsibleExitJson) as idempotent_result: self.module.main() idempotent_exc = idempotent_result.exception.args[0] # Ensure both calls resulted in the same returned value # except for changed which should be false the second time self.assertFalse(idempotent_exc["changed"]) self.assertEqual(idempotent_exc["response"], delete_exc["response"]) def test_absent_with_existing_users(self): """Test organization removal with attached users should fail except if force is true.""" module_args = { "state": "absent", "pritunl_api_token": "token", "pritunl_api_secret": "secret", "pritunl_url": "https://pritunl.domain.com", "name": "GumGum", } set_module_args(module_args) # Test deletion with self.patch_get_pritunl_organizations( side_effect=PritunlListOrganizationMock ) as mock_get: with self.patch_delete_pritunl_organization( side_effect=PritunlDeleteOrganizationMock ) as mock_delete: with self.assertRaises(AnsibleFailJson) as failure_result: self.module.main() failure_exc = failure_result.exception.args[0] self.assertRegex(failure_exc["msg"], "Can not remove organization") # Switch force=True which should run successfully set_module_args(dict_merge(module_args, {"force": True})) with self.patch_get_pritunl_organizations( side_effect=PritunlListOrganizationMock ) as mock_get: with self.patch_delete_pritunl_organization( side_effect=PritunlDeleteOrganizationMock ) as mock_delete: with self.assertRaises(AnsibleExitJson) as delete_result: self.module.main() delete_exc = delete_result.exception.args[0] self.assertTrue(delete_exc["changed"])