1 # Copyright 2013: Mirantis Inc.
4 # Licensed under the Apache License, Version 2.0 (the "License"); you may
5 # not use this file except in compliance with the License. You may obtain
6 # a copy of the License at
8 # http://www.apache.org/licenses/LICENSE-2.0
10 # Unless required by applicable law or agreed to in writing, software
11 # distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
12 # WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
13 # License for the specific language governing permissions and limitations
16 # yardstick comment: this is a modified copy of rally/rally/common/utils.py
18 from __future__ import absolute_import
19 from __future__ import print_function
30 from functools import reduce
31 from contextlib import closing
35 from flask import jsonify
36 from six.moves import configparser
37 from oslo_utils import importutils
38 from oslo_serialization import jsonutils
42 logger = logging.getLogger(__name__)
43 logger.setLevel(logging.DEBUG)
46 # Decorator for cli-args
47 def cliargs(*args, **kwargs):
49 func.__dict__.setdefault('arguments', []).insert(0, (args, kwargs))
54 def itersubclasses(cls, _seen=None):
55 """Generator over all subclasses of a given class in depth first order."""
57 if not isinstance(cls, type):
58 raise TypeError("itersubclasses must be called with "
59 "new-style classes, not %.100r" % cls)
60 _seen = _seen or set()
62 subs = cls.__subclasses__()
63 except TypeError: # fails only when cls is type
64 subs = cls.__subclasses__(cls)
69 for sub in itersubclasses(sub, _seen):
73 def try_append_module(name, modules):
74 if name not in modules:
75 modules[name] = importutils.import_module(name)
78 def import_modules_from_package(package):
79 """Import modules from package and append into sys.modules
81 :param: package - Full package name. For example: rally.deploy.engines
83 path = [os.path.dirname(yardstick.__file__), ".."] + package.split(".")
84 path = os.path.join(*path)
85 for root, dirs, files in os.walk(path):
86 for filename in files:
87 if filename.startswith("__") or not filename.endswith(".py"):
89 new_package = ".".join(root.split(os.sep)).split("....")[1]
90 module_name = "%s.%s" % (new_package, filename[:-3])
92 try_append_module(module_name, sys.modules)
94 logger.exception("unable to import %s", module_name)
97 def parse_yaml(file_path):
99 with open(file_path) as f:
100 value = yaml.safe_load(f)
104 if e.errno != errno.EEXIST:
110 def get_param(key, default=''):
112 conf_file = os.environ.get('CONF_FILE', '/etc/yardstick/yardstick.yaml')
114 conf = parse_yaml(conf_file)
116 return reduce(lambda a, b: a[b], key.split('.'), conf)
127 if e.errno != errno.EEXIST:
131 def remove_file(path):
135 if e.errno != errno.ENOENT:
139 def execute_command(cmd):
140 exec_msg = "Executing command: '%s'" % cmd
141 logger.debug(exec_msg)
143 output = subprocess.check_output(cmd.split()).split(os.linesep)
148 def source_env(env_file):
149 p = subprocess.Popen(". %s; env" % env_file, stdout=subprocess.PIPE,
151 output = p.communicate()[0]
152 env = dict(line.split('=', 1) for line in output.splitlines() if '=' in line)
153 os.environ.update(env)
157 def read_json_from_file(path):
158 with open(path, 'r') as f:
160 # don't use jsonutils.load() it conflicts with already decoded input
161 return jsonutils.loads(j)
164 def write_json_to_file(path, data, mode='w'):
165 with open(path, mode) as f:
166 jsonutils.dump(data, f)
169 def write_file(path, data, mode='w'):
170 with open(path, mode) as f:
174 def parse_ini_file(path):
175 parser = configparser.ConfigParser()
178 files = parser.read(path)
179 except configparser.MissingSectionHeaderError:
180 logger.exception('invalid file type')
184 raise RuntimeError('file not exist')
187 default = {k: v for k, v in parser.items('DEFAULT')}
188 except configparser.NoSectionError:
191 config = dict(DEFAULT=default,
192 **{s: {k: v for k, v in parser.items(
193 s)} for s in parser.sections()})
198 def get_port_mac(sshclient, port):
199 cmd = "ifconfig |grep HWaddr |grep %s |awk '{print $5}' " % port
200 status, stdout, stderr = sshclient.execute(cmd)
203 raise RuntimeError(stderr)
204 return stdout.rstrip()
207 def get_port_ip(sshclient, port):
208 cmd = "ifconfig %s |grep 'inet addr' |awk '{print $2}' " \
209 "|cut -d ':' -f2 " % port
210 status, stdout, stderr = sshclient.execute(cmd)
213 raise RuntimeError(stderr)
214 return stdout.rstrip()
217 def flatten_dict_key(data):
220 # use list, because iterable is too generic
221 if not any(isinstance(v, (collections.Mapping, list))
222 for v in data.values()):
225 for k, v in data.items():
226 if isinstance(v, collections.Mapping):
227 for n_k, n_v in v.items():
228 next_data["%s.%s" % (k, n_k)] = n_v
229 # use list because iterable is too generic
230 elif isinstance(v, collections.Iterable) and not isinstance(v, six.string_types):
231 for index, item in enumerate(v):
232 next_data["%s%d" % (k, index)] = item
236 return flatten_dict_key(next_data)
239 def translate_to_str(obj):
240 if isinstance(obj, collections.Mapping):
241 return {str(k): translate_to_str(v) for k, v in obj.items()}
242 elif isinstance(obj, list):
243 return [translate_to_str(ele) for ele in obj]
244 elif isinstance(obj, six.text_type):
249 def result_handler(status, data):
254 return jsonify(result)
257 def change_obj_to_dict(obj):
259 for k, v in vars(obj).items():
267 def set_dict_value(dic, keys, value):
270 for key in keys.split('.'):
271 return_dic.setdefault(key, {})
272 if key == keys.split('.')[-1]:
273 return_dic[key] = value
275 return_dic = return_dic[key]
279 def get_free_port(ip):
280 with closing(socket.socket(socket.AF_INET, socket.SOCK_STREAM)) as s:
282 port = random.randint(5000, 10000)
283 if s.connect_ex((ip, port)) != 0:
287 def mac_address_to_hex_list(mac):
288 octets = ["0x{:02x}".format(int(elem, 16)) for elem in mac.split(':')]
289 assert len(octets) == 6 and all(len(octet) == 4 for octet in octets)
293 def safe_ip_address(ip_addr):
294 """ get ip address version v6 or v4 """
296 return ipaddress.ip_address(six.text_type(ip_addr))
298 logging.error("%s is not valid", ip_addr)
302 def get_ip_version(ip_addr):
303 """ get ip address version v6 or v4 """
305 address = ipaddress.ip_address(six.text_type(ip_addr))
307 logging.error("%s is not valid", ip_addr)
310 return address.version
313 def ip_to_hex(ip_addr):
315 address = ipaddress.ip_address(six.text_type(ip_addr))
317 logging.error("%s is not valid", ip_addr)
320 if address.version != 4:
322 return '{:08x}'.format(int(address))
325 def try_int(s, *args):
326 """Convert to integer if possible."""
329 except (TypeError, ValueError):
330 return args[0] if args else s
333 class SocketTopology(dict):
336 return sorted(self.keys())
339 return sorted(core for cores in self.values() for core in cores)
341 def processors(self):
343 proc for cores in self.values() for procs in cores.values() for
347 def parse_cpuinfo(cpuinfo):
350 lines = cpuinfo.splitlines()
356 name, value = line.split(":", 1)
357 core_lines[name.strip()] = try_int(value.strip())
359 core_details.append(core_lines)
362 for core in core_details:
363 socket_map.setdefault(core["physical id"], {}).setdefault(
364 core["core id"], {})[core["processor"]] = (
365 core["processor"], core["core id"], core["physical id"])
367 return SocketTopology(socket_map)
370 def config_to_dict(config):
371 return {section: dict(config.items(section)) for section in
375 def validate_non_string_sequence(value, default=None, raise_exc=None):
376 if isinstance(value, collections.Sequence) and not isinstance(value, str):
383 def join_non_strings(separator, *non_strings):
385 non_strings = validate_non_string_sequence(non_strings[0], raise_exc=RuntimeError)
386 except (IndexError, RuntimeError):
388 return str(separator).join(str(non_string) for non_string in non_strings)
391 class ErrorClass(object):
393 def __init__(self, *args, **kwargs):
394 if 'test' not in kwargs:
397 def __getattr__(self, item):