Clone of official tools

Revision:
0:66f3b5499f7f
Child:
7:5af61d55adbe
diff -r 000000000000 -r 66f3b5499f7f utils.py
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/utils.py	Thu May 19 19:44:41 2016 +0100
@@ -0,0 +1,176 @@
+"""
+mbed SDK
+Copyright (c) 2011-2013 ARM Limited
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+    http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+"""
+import sys
+import inspect
+import os
+from os import listdir, remove, makedirs
+from shutil import copyfile
+from os.path import isdir, join, exists, split, relpath, splitext
+from subprocess import Popen, PIPE, STDOUT, call
+
+
+def cmd(l, check=True, verbose=False, shell=False, cwd=None):
+    text = l if shell else ' '.join(l)
+    if verbose:
+        print text
+    rc = call(l, shell=shell, cwd=cwd)
+    if check and rc != 0:
+        raise Exception('ERROR %d: "%s"' % (rc, text))
+
+
+def run_cmd(command, wd=None, redirect=False):
+    assert is_cmd_valid(command[0])
+    try:
+        p = Popen(command, stdout=PIPE, stderr=STDOUT if redirect else PIPE, cwd=wd)
+        _stdout, _stderr = p.communicate()
+    except:
+        print "[OS ERROR] Command: "+(' '.join(command))
+        raise
+    return _stdout, _stderr, p.returncode
+
+
+def run_cmd_ext(command):
+    assert is_cmd_valid(command[0])
+    p = Popen(command, stdout=PIPE, stderr=PIPE)
+    _stdout, _stderr = p.communicate()
+    return _stdout, _stderr, p.returncode
+
+
+def is_cmd_valid(cmd):
+    caller = get_caller_name()
+    abspath = find_cmd_abspath(cmd)
+    if not abspath:
+        error("%s: Command '%s' can't be found" % (caller, cmd))
+    if not is_exec(abspath):
+        error("%s: Command '%s' resolves to file '%s' which is not executable" % (caller, cmd, abspath))
+    return True
+
+
+def is_exec(path):
+    return os.access(path, os.X_OK) or os.access(path+'.exe', os.X_OK)
+
+
+def find_cmd_abspath(cmd):
+    """ Returns the absolute path to a command.
+        None is returned if no absolute path was found.
+    """
+    if exists(cmd) or exists(cmd + '.exe'):
+        return os.path.abspath(cmd)
+    if not 'PATH' in os.environ:
+        raise Exception("Can't find command path for current platform ('%s')" % sys.platform)
+    PATH=os.environ['PATH']
+    for path in PATH.split(os.pathsep):
+        abspath = '%s/%s' % (path, cmd)
+        if exists(abspath) or exists(abspath + '.exe'):
+            return abspath
+
+
+def mkdir(path):
+    if not exists(path):
+        makedirs(path)
+
+
+def copy_file(src, dst):
+    """ Implement the behaviour of "shutil.copy(src, dst)" without copying the
+        permissions (this was causing errors with directories mounted with samba)
+    """
+    if isdir(dst):
+        _, file = split(src)
+        dst = join(dst, file)
+    copyfile(src, dst)
+
+
+def delete_dir_files(dir):
+    if not exists(dir):
+        return
+
+    for f in listdir(dir):
+        file = join(dir, f)
+        if not isdir(file):
+            remove(file)
+
+
+def get_caller_name(steps=2):
+    """
+    When called inside a function, it returns the name
+    of the caller of that function.
+    """
+    return inspect.stack()[steps][3]
+
+
+def error(msg):
+    print("ERROR: %s" % msg)
+    sys.exit(1)
+
+
+def rel_path(path, base, dot=False):
+    p = relpath(path, base)
+    if dot and not p.startswith('.'):
+        p = './' + p
+    return p
+
+
+class ToolException(Exception):
+    pass
+
+class NotSupportedException(Exception):
+    pass
+
+def split_path(path):
+    base, file = split(path)
+    name, ext = splitext(file)
+    return base, name, ext
+
+
+def args_error(parser, message):
+    print "\n\n%s\n\n" % message
+    parser.print_help()
+    sys.exit()
+
+
+def construct_enum(**enums):
+    """ Create your own pseudo-enums """
+    return type('Enum', (), enums)
+
+
+def check_required_modules(required_modules, verbose=True):
+    """ Function checks for Python modules which should be "importable" (installed)
+        before test suite can be used.
+        @return returns True if all modules are installed already
+    """
+    import imp
+    not_installed_modules = []
+    for module_name in required_modules:
+        try:
+            imp.find_module(module_name)
+        except ImportError as e:
+            # We also test against a rare case: module is an egg file
+            try:
+                __import__(module_name)
+            except ImportError as e:
+                not_installed_modules.append(module_name)
+                if verbose:
+                    print "Error: %s" % e
+
+    if verbose:
+        if not_installed_modules:
+            print "Warning: Module(s) %s not installed. Please install required module(s) before using this script."% (', '.join(not_installed_modules))
+
+    if not_installed_modules:
+        return False
+    else:
+        return True