diff options
| author | zsdc <taras@vyos.io> | 2022-03-25 20:58:01 +0200 | 
|---|---|---|
| committer | zsdc <taras@vyos.io> | 2022-03-25 21:42:00 +0200 | 
| commit | 31448cccedd8f841fb3ac7d0f2e3cdefe08a53ba (patch) | |
| tree | 349631a02467dae0158f6f663cc8aa8537974a97 /tests/unittests/config/test_cc_runcmd.py | |
| parent | 5c4b3943343a85fbe517e5ec1fc670b3a8566b4b (diff) | |
| parent | 8537237d80a48c8f0cbf8e66aa4826bbc882b022 (diff) | |
| download | vyos-cloud-init-31448cccedd8f841fb3ac7d0f2e3cdefe08a53ba.tar.gz vyos-cloud-init-31448cccedd8f841fb3ac7d0f2e3cdefe08a53ba.zip | |
T2117: Cloud-init updated to 22.1
Merged with 22.1 tag from the upstream Cloud-init repository.
Our modules were slightly modified for compatibility with the new
version.
Diffstat (limited to 'tests/unittests/config/test_cc_runcmd.py')
| -rw-r--r-- | tests/unittests/config/test_cc_runcmd.py | 137 | 
1 files changed, 137 insertions, 0 deletions
| diff --git a/tests/unittests/config/test_cc_runcmd.py b/tests/unittests/config/test_cc_runcmd.py new file mode 100644 index 00000000..59490d67 --- /dev/null +++ b/tests/unittests/config/test_cc_runcmd.py @@ -0,0 +1,137 @@ +# This file is part of cloud-init. See LICENSE file for license information. +import logging +import os +import stat +from unittest.mock import patch + +from cloudinit import helpers, subp, util +from cloudinit.config.cc_runcmd import handle, schema +from tests.unittests.helpers import ( +    CiTestCase, +    FilesystemMockingTestCase, +    SchemaTestCaseMixin, +    skipUnlessJsonSchema, +) +from tests.unittests.util import get_cloud + +LOG = logging.getLogger(__name__) + + +class TestRuncmd(FilesystemMockingTestCase): + +    with_logs = True + +    def setUp(self): +        super(TestRuncmd, self).setUp() +        self.subp = subp.subp +        self.new_root = self.tmp_dir() +        self.patchUtils(self.new_root) +        self.paths = helpers.Paths({"scripts": self.new_root}) + +    def test_handler_skip_if_no_runcmd(self): +        """When the provided config doesn't contain runcmd, skip it.""" +        cfg = {} +        mycloud = get_cloud(paths=self.paths) +        handle("notimportant", cfg, mycloud, LOG, None) +        self.assertIn( +            "Skipping module named notimportant, no 'runcmd' key", +            self.logs.getvalue(), +        ) + +    @patch("cloudinit.util.shellify") +    def test_runcmd_shellify_fails(self, cls): +        """When shellify fails throw exception""" +        cls.side_effect = TypeError("patched shellify") +        valid_config = {"runcmd": ["echo 42"]} +        cc = get_cloud(paths=self.paths) +        with self.assertRaises(TypeError) as cm: +            with self.allow_subp(["/bin/sh"]): +                handle("cc_runcmd", valid_config, cc, LOG, None) +        self.assertIn("Failed to shellify", str(cm.exception)) + +    def test_handler_invalid_command_set(self): +        """Commands which can't be converted to shell will raise errors.""" +        invalid_config = {"runcmd": 1} +        cc = get_cloud(paths=self.paths) +        with self.assertRaises(TypeError) as cm: +            handle("cc_runcmd", invalid_config, cc, LOG, []) +        self.assertIn( +            "Failed to shellify 1 into file" +            " /var/lib/cloud/instances/iid-datasource-none/scripts/runcmd", +            str(cm.exception), +        ) + +    @skipUnlessJsonSchema() +    def test_handler_schema_validation_warns_non_array_type(self): +        """Schema validation warns of non-array type for runcmd key. + +        Schema validation is not strict, so runcmd attempts to shellify the +        invalid content. +        """ +        invalid_config = {"runcmd": 1} +        cc = get_cloud(paths=self.paths) +        with self.assertRaises(TypeError) as cm: +            handle("cc_runcmd", invalid_config, cc, LOG, []) +        self.assertIn( +            "Invalid cloud-config provided:\nruncmd: 1 is not of type 'array'", +            self.logs.getvalue(), +        ) +        self.assertIn("Failed to shellify", str(cm.exception)) + +    @skipUnlessJsonSchema() +    def test_handler_schema_validation_warns_non_array_item_type(self): +        """Schema validation warns of non-array or string runcmd items. + +        Schema validation is not strict, so runcmd attempts to shellify the +        invalid content. +        """ +        invalid_config = { +            "runcmd": ["ls /", 20, ["wget", "http://stuff/blah"], {"a": "n"}] +        } +        cc = get_cloud(paths=self.paths) +        with self.assertRaises(TypeError) as cm: +            handle("cc_runcmd", invalid_config, cc, LOG, []) +        expected_warnings = [ +            "runcmd.1: 20 is not valid under any of the given schemas", +            "runcmd.3: {'a': 'n'} is not valid under any of the given schema", +        ] +        logs = self.logs.getvalue() +        for warning in expected_warnings: +            self.assertIn(warning, logs) +        self.assertIn("Failed to shellify", str(cm.exception)) + +    def test_handler_write_valid_runcmd_schema_to_file(self): +        """Valid runcmd schema is written to a runcmd shell script.""" +        valid_config = {"runcmd": [["ls", "/"]]} +        cc = get_cloud(paths=self.paths) +        handle("cc_runcmd", valid_config, cc, LOG, []) +        runcmd_file = os.path.join( +            self.new_root, +            "var/lib/cloud/instances/iid-datasource-none/scripts/runcmd", +        ) +        self.assertEqual("#!/bin/sh\n'ls' '/'\n", util.load_file(runcmd_file)) +        file_stat = os.stat(runcmd_file) +        self.assertEqual(0o700, stat.S_IMODE(file_stat.st_mode)) + + +@skipUnlessJsonSchema() +class TestSchema(CiTestCase, SchemaTestCaseMixin): +    """Directly test schema rather than through handle.""" + +    schema = schema + +    def test_duplicates_are_fine_array_array(self): +        """Duplicated commands array/array entries are allowed.""" +        self.assertSchemaValid( +            [["echo", "bye"], ["echo", "bye"]], +            "command entries can be duplicate.", +        ) + +    def test_duplicates_are_fine_array_string(self): +        """Duplicated commands array/string entries are allowed.""" +        self.assertSchemaValid( +            ["echo bye", "echo bye"], "command entries can be duplicate." +        ) + + +# vi: ts=4 expandtab | 
