[Asterisk-code-review] testsuite: Allow copying of arbitrary files to asterisk instances (testsuite[master])

Friendly Automation asteriskteam at digium.com
Wed Oct 6 09:09:46 CDT 2021


Friendly Automation has submitted this change. ( https://gerrit.asterisk.org/c/testsuite/+/16588 )

Change subject: testsuite: Allow copying of arbitrary files to asterisk instances
......................................................................

testsuite: Allow copying of arbitrary files to asterisk instances

If you have a key or certificate file that has to go in
/var/lib/asterisk/keys, there was no way for a test to get
that file into that directory.  Various work-arounds ensued.

* Added a replaceable parameter "<<instanceid>>" to config
  file processing which would be replaced with the Asterisk
  instance's id.  I.E.  'ast1' would be '1'.

* Added automatic installation of files in a test's "files"
  directory to its Asterisk instance's well-known directories.
  To use, create directories named after entries in the
  asterisk.conf "directories" category and place your files there.
  It works much like the test's "configs" directory.

  For example:
    mytest/
      configs/
        ast1/
          someconfig.conf
        ast2/
          someconfig.conf
      files/
        common/
          astvarlibdir/
            keys/
              ca.crt
        ast1/
          astvarlibdir/
            keys/
              mykey1.pem
        ast2/
          astvarlibdir/
            keys/
              mykey2.pem
      test-config.yaml

  Since 'astvarlibdir' is usually defined in asterisk.conf as
  '/var/lib/asterisk', this would copy 'ca.crt' to both Asterisk
  instance's '/var/lib/asterisk/keys' directory, 'mykey1.pem' to
  the first Asterisk instance's '/var/lib/asterisk/keys' directory
  and 'mykey2.pem' to the second instance's '/var/lib/asterisk/keys'
  directory.

  Each of the instance's "someconfig.conf" could reference those
  files like so:

  cacert = <<astvarlibdir>>/keys/ca.crt.pem
  keyfile = <<astvarlibdir>>/keys/mykey<<instanceid>>.pem

  If you have files that can be shared among tests, you can define
  'base-files-path' in each test's 'test-object-config' section which
  would point to a directory structured like the 'files' directory
  above.

  For example
    mytests/
      myfiles/
        common/
          astvarlibdir/
            keys/
              ca.crt
        ast1/
          astvarlibdir/
            keys/
              mykey1.pem
        ast2/
          astvarlibdir/
            keys/
              mykey2.pem

      mytest1/
        ...
      mytest2/
        ...

    Then in both test's test-config.yaml...
    test-object-config:
        base-files-path: 'tests/mytests/myfiles'

  base-files-path is processed before the automatic files
  processing of each test so individual tests can add its own files
  or even override a common file if it chooses.

ASTERISK-29675

Change-Id: I69bb54fea459e83f65a474d3b74c40b28fe01b4c
---
M README.txt
M lib/python/asterisk/asterisk.py
M lib/python/asterisk/test_case.py
A lib/python/polyfill.py
M sample-yaml/test-config.yaml.sample
5 files changed, 221 insertions(+), 2 deletions(-)

Approvals:
  George Joseph: Looks good to me, approved
  Friendly Automation: Approved for Submit



diff --git a/README.txt b/README.txt
index 459fbed..93888be 100644
--- a/README.txt
+++ b/README.txt
@@ -335,6 +335,41 @@
 Since each Asterisk instance required difference SIP settings, each 'ast%d'
 folder will have a different sip.conf file.
 
+You can also copy arbitrary files like sound, key and certificate files into
+any of the entries in the asterisk.conf "directories" category using its
+entry name name.  All intermediate directories will be created if they
+don't already exist.
+
+For example, to have a certificate and keys copied to an Asterisk instance's
+'/var/lib/asterisk/keys' directory, you'd place it in...
+
+    basic-call/
+        files/
+            common/
+                astvarlibdir/
+                    keys/
+                        ca.crt
+            ast1/
+                astvarlibdir/
+                    keys/
+                        instance1-key.pem
+                ...
+            ast2/
+                astvarlibdir/
+                    keys/
+                        instance2-key.pem
+
+Since 'astvarlibdir' is defined in asterisk.conf as '/var/lib/asterisk',
+this would copy 'ca.crt' to both instance's '/var/lib/asterisk/keys/' directory,
+'instance1-key.pem' to instance 1's '/var/lib/asterisk/keys/' directory and
+'instance2-key.pem' to instance 2's '/var/lib/asterisk/keys/' directory.
+
+If you have files that can be shared among multiple tests, you can create a
+directory following the same structure as above in some parent directory and
+direct each test to include it with the 'base-files-path' parameter in
+its test-config.yaml file.  See sample-yaml/test-config.yaml.sample for
+more info.
+
 d) Test Execution
 
         The "run-test" executable will be run by a top level application in the
diff --git a/lib/python/asterisk/asterisk.py b/lib/python/asterisk/asterisk.py
index 977dc6e..f1371db 100644
--- a/lib/python/asterisk/asterisk.py
+++ b/lib/python/asterisk/asterisk.py
@@ -16,6 +16,7 @@
 import shutil
 import logging
 import fileinput
+import polyfill
 
 from . import test_suite_utils
 
@@ -43,7 +44,6 @@
 
 LOGGER = logging.getLogger(__name__)
 
-
 class AsteriskRemoteProtocol(protocol.Protocol):
     """Class that acts as a remote protocol to Asterisk"""
 
@@ -367,6 +367,7 @@
         self.original_astmoddir = ""
         self.remote_config = remote_config
         self.memcheck_delay_stop = 0
+        self.instance_id = 0
         if test_config is not None and 'memcheck-delay-stop' in test_config:
             self.memcheck_delay_stop = test_config['memcheck-delay-stop'] or 0
 
@@ -419,6 +420,7 @@
             while True:
                 if not os.path.isdir("%s/ast%d" % (self.base, i)):
                     self.base = "%s/ast%d" % (self.base, i)
+                    self.instance_id = i;
                     break
                 i += 1
 
@@ -673,6 +675,7 @@
         for key in self.directories.keys():
             value = value.replace("<<%s>>" % key,
                               "%s%s" % (self.base, self.directories[key]))
+            value = value.replace("<<instanceid>>", self.instance_id)
         return value
 
 # Quick little function for doing search and replace in a file used below.
@@ -787,6 +790,61 @@
         except IOError:
             LOGGER.warn("The destination is not writable '%s'" % target_path)
 
+    def install_files(self, source_path):
+        """Installs all files located in a directory to this
+        instance of Asterisk.
+
+        Keyword Arguments:
+        source_path This argument must be the path to the directory
+                 containing the files to be installed into standard
+                 locations.
+
+                 Example: tests/my-cool-test/files/ast1
+                 This directory must contain at least one sub-directory
+                 matching the name of an entry from the asterisk.conf
+                 "directories" category.  For example "astvarlibdir".
+                 Each entry in those directories will be copied to the
+                 runtime equivalent for the test.
+
+        Example:
+            Given the following filesystem layout:
+            tests/my-cool-test/files/ast1
+                                    - astvarlibdir/
+                                        - sounds/
+                                            - recordings/
+                                                - myrecording.ulaw
+                                    - astkeydir/keys/
+                                        - mykey.pem
+
+            ...and given the following entries in asterisk.conf:
+            astvarlibdir => /var/lib/asterisk
+            astkeydir => /var/lib/asterisk
+
+            The myrecording.ulaw and mykey.pem files would be copied to
+            <testroot>/var/lib/asterisk/sounds/recordings/ and
+            <testroot>/var/lib/asterisk/keys/ respectively.
+        """
+
+        self._make_directory_structure()
+        if not os.path.exists(source_path):
+            return
+
+        for fname in os.listdir(source_path):
+            source = "%s/%s" % (source_path, fname)
+
+            if os.path.isdir(source):
+                # If the directory name isn't one of the
+                # well known directories, just skip it.
+                if not self.directories[fname]:
+                    continue
+                # join() doesn't work if one of the paths is absolute
+                # so we have to skip the leading '/' in the directory
+                # entry.  I.E /var/lib/asterisk becomes var/lib/asterisk
+                direntry = self.directories[fname]
+                dirpath = direntry[1:] if direntry[0] == '/' else direntry
+                dest = os.path.join(self.base, dirpath)
+                polyfill.copytree(source, dest, dirs_exist_ok=True)
+
     def _overwrite_file(self, filename, values):
         """Overwrite a particular config file
 
diff --git a/lib/python/asterisk/test_case.py b/lib/python/asterisk/test_case.py
index 1c7edfa..c4cde27 100644
--- a/lib/python/asterisk/test_case.py
+++ b/lib/python/asterisk/test_case.py
@@ -122,6 +122,7 @@
         self.ami = []
         self.fastagi = []
         self.base_config_path = None
+        self.base_files_path = None
         self.reactor_timeout = 30
         self.passed = None
         self.fail_tokens = []
@@ -149,6 +150,8 @@
         if test_config:
             if 'config-path' in test_config:
                 self.base_config_path = test_config['config-path']
+            if 'base-files-path' in test_config:
+                self.base_files_path = test_config['base-files-path']
             if 'reactor-timeout' in test_config:
                 self.reactor_timeout = test_config['reactor-timeout']
             if 'memcheck-delay-stop' in test_config:
@@ -248,7 +251,8 @@
                          for i in range(count)]
         return asterisks
 
-    def create_asterisk(self, count=1, base_configs_path=None, test_config=None):
+    def create_asterisk(self, count=1, base_configs_path=None, test_config=None,
+                        base_files_path=None):
         """Create n instances of Asterisk
 
         Note: if the instances of Asterisk being created are remote, the
@@ -264,6 +268,11 @@
                           configuration can be overwritten by individual tests,
                           however.
         test_config       Test Configuration
+        base_files_path   Provides common files for Asterisk instances
+                          to use. This is useful for certain test types that use
+                          the same files, like keys, all the time. This
+                          configuration can be overwritten by individual tests,
+                          however.
         """
         for i, ast_config in enumerate(self.get_asterisk_hosts(count)):
             local_num = ast_config.get('num')
@@ -300,6 +309,19 @@
                                             (self.test_name, local_num),
                                             self.test_config.get_deps())
 
+                # If a base files directory exists for this Asterisk instance
+                # has been provided, install it first
+                if base_files_path is None:
+                    base_files_path = self.base_files_path
+                if base_files_path:
+                    self.ast[i].install_files("%s/common" % (base_files_path))
+                    self.ast[i].install_files("%s/ast%d" %
+                                            (base_files_path, local_num))
+                # Copy test specific config files
+                self.ast[i].install_files("%s/files/common" % (self.test_name))
+                self.ast[i].install_files("%s/files/ast%d" %
+                                            (self.test_name, local_num))
+
     def create_ami_factory(self, count=1, username="user", secret="mysecret",
                            port=5038):
         """Create n instances of AMI.  Each AMI instance will attempt to connect
diff --git a/lib/python/polyfill.py b/lib/python/polyfill.py
new file mode 100644
index 0000000..bd95d5f
--- /dev/null
+++ b/lib/python/polyfill.py
@@ -0,0 +1,85 @@
+import sys
+import os
+import shutil
+
+"""
+    These are polyfills for a few file/directory functions
+    that don't show up until Python 3.9.
+"""
+
+def makedirs(name, mode=0o777, exist_ok=False):
+    head, tail = os.path.split(name)
+    if not tail:
+        head, tail = os.path.split(head)
+    if head and tail and not os.path.exists(head):
+        try:
+            makedirs(head, exist_ok=exist_ok)
+        except FileExistsError:
+            # Defeats race condition when another thread created the path
+            pass
+        cdir = os.curdir
+        if isinstance(tail, bytes):
+            cdir = bytes(os.curdir, 'ASCII')
+        if tail == cdir:           # xxx/newdir/. exists if xxx/newdir exists
+            return
+    try:
+        os.mkdir(name, mode)
+    except OSError:
+        if not exist_ok or not os.path.isdir(name):
+            raise
+
+def copytree(src, dst, symlinks=False, ignore=None, copy_function=shutil.copy2,
+             ignore_dangling_symlinks=False, dirs_exist_ok=False):
+    names = os.listdir(src)
+    if ignore is not None:
+        ignored_names = ignore(src, names)
+    else:
+        ignored_names = set()
+
+    makedirs(dst, exist_ok=dirs_exist_ok)
+    errors = []
+    for name in names:
+        if name in ignored_names:
+            continue
+        srcname = os.path.join(src, name)
+        dstname = os.path.join(dst, name)
+        try:
+            if os.path.islink(srcname):
+                linkto = os.readlink(srcname)
+                if symlinks:
+                    # We can't just leave it to `copy_function` because legacy
+                    # code with a custom `copy_function` may rely on copytree
+                    # doing the right thing.
+                    os.symlink(linkto, dstname)
+                    shutil.copystat(srcname, dstname, follow_symlinks=not symlinks)
+                else:
+                    # ignore dangling symlink if the flag is on
+                    if not os.path.exists(linkto) and ignore_dangling_symlinks:
+                        continue
+                    # otherwise let the copy occurs. copy2 will raise an error
+                    if os.path.isdir(srcname):
+                        copytree(srcname, dstname, symlinks, ignore,
+                                 copy_function, dirs_exist_ok=dirs_exist_ok)
+                    else:
+                        copy_function(srcname, dstname)
+            elif os.path.isdir(srcname):
+                copytree(srcname, dstname, symlinks, ignore, copy_function, dirs_exist_ok=dirs_exist_ok)
+            else:
+                # Will raise a SpecialFileError for unsupported file types
+                copy_function(srcname, dstname)
+        # catch the Error from the recursive copytree so that we can
+        # continue with other files
+        except shutil.Error as err:
+            errors.extend(err.args[0])
+        except OSError as why:
+            errors.append((srcname, dstname, str(why)))
+    try:
+        shutil.copystat(src, dst)
+    except OSError as why:
+        # Copying file access times may fail on Windows
+        if getattr(why, 'winerror', None) is None:
+            errors.append((src, dst, str(why)))
+    if errors:
+        raise shutil.Error(errors)
+    return dst
+
diff --git a/sample-yaml/test-config.yaml.sample b/sample-yaml/test-config.yaml.sample
index 773b98f..37a2de9 100644
--- a/sample-yaml/test-config.yaml.sample
+++ b/sample-yaml/test-config.yaml.sample
@@ -204,6 +204,25 @@
     # large sets of tests that cover the same functionality, and allows them
     # to share config files.
     config-path: 'tests/foo'
+
+    # A path to a base directory containing abritrary files to be copied
+    # into the Asterisk well-known directories for Asterisk instances
+    # in this test.
+    # The directory should be relative to the testsuite top-level directory.
+    # For example... If set to 'tests/mytests/myfiles' then files in
+    # 'tests/mytest/myfiles/common/astvarlibdir/keys' would be copied to all
+    # of this test's Asterisk instances's '/var/lib/asterisk/keys' directories.
+    # Files in 'tests/mytest/myfiles/ast1/astvarlibdir/keys' would be copied
+    # into the first Asterisk instance's '/var/lib/asterisk/keys' directory.
+    # Files in 'tests/mytest/myfiles/ast2/astvarlibdir/keys' would be copied
+    # into the second Asterisk instance's '/var/lib/asterisk/keys' directory.
+    # etc.
+    # This is useful for tests that can share files like keys and certificates.
+    # NOTE: Files in individual '<testdir>/files' directories are automatically
+    # copied so you don't need this parameter to share files among instances
+    # in the same test.
+    base-files-directory: 'tests/mytests/myfiles'
+
     reactor-timeout: 30
     spawn-after-hangup: True
     # Indicate that the test is going to be restarting Asterisk so

-- 
To view, visit https://gerrit.asterisk.org/c/testsuite/+/16588
To unsubscribe, or for help writing mail filters, visit https://gerrit.asterisk.org/settings

Gerrit-Project: testsuite
Gerrit-Branch: master
Gerrit-Change-Id: I69bb54fea459e83f65a474d3b74c40b28fe01b4c
Gerrit-Change-Number: 16588
Gerrit-PatchSet: 1
Gerrit-Owner: George Joseph <gjoseph at digium.com>
Gerrit-Reviewer: Friendly Automation
Gerrit-Reviewer: George Joseph <gjoseph at digium.com>
Gerrit-MessageType: merged
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://lists.digium.com/pipermail/asterisk-code-review/attachments/20211006/aa4e72da/attachment-0001.html>


More information about the asterisk-code-review mailing list