1#!/usr/bin/env python3 2 3r""" 4This module provides many print functions such as sprint_var, sprint_time, sprint_error, sprint_call_stack. 5""" 6 7import sys 8import os 9import time 10import inspect 11import re 12import grp 13import socket 14import argparse 15import copy 16try: 17 import __builtin__ 18except ImportError: 19 import builtins as __builtin__ 20import logging 21import collections 22from wrap_utils import * 23 24try: 25 robot_env = 1 26 from robot.utils import DotDict 27 from robot.utils import NormalizedDict 28 from robot.libraries.BuiltIn import BuiltIn 29 # Having access to the robot libraries alone does not indicate that we are in a robot environment. The 30 # following try block should confirm that. 31 try: 32 var_value = BuiltIn().get_variable_value("${SUITE_NAME}", "") 33 except BaseException: 34 robot_env = 0 35except ImportError: 36 robot_env = 0 37 38import gen_arg as ga 39 40# Setting these variables for use both inside this module and by programs importing this module. 41pgm_file_path = sys.argv[0] 42pgm_name = os.path.basename(pgm_file_path) 43pgm_dir_path = os.path.normpath(re.sub("/" + pgm_name, "", pgm_file_path)) +\ 44 os.path.sep 45 46 47# Some functions (e.g. sprint_pgm_header) have need of a program name value that looks more like a valid 48# variable name. Therefore, we'll swap odd characters like "." out for underscores. 49pgm_name_var_name = pgm_name.replace(".", "_") 50 51# Initialize global values used as defaults by print_time, print_var, etc. 52dft_indent = 0 53 54# Calculate default column width for print_var functions based on environment variable settings. The 55# objective is to make the variable values line up nicely with the time stamps. 56dft_col1_width = 29 57 58NANOSECONDS = os.environ.get('NANOSECONDS', '1') 59 60if NANOSECONDS == "1": 61 dft_col1_width = dft_col1_width + 7 62 63SHOW_ELAPSED_TIME = os.environ.get('SHOW_ELAPSED_TIME', '1') 64 65if SHOW_ELAPSED_TIME == "1": 66 if NANOSECONDS == "1": 67 dft_col1_width = dft_col1_width + 14 68 else: 69 dft_col1_width = dft_col1_width + 7 70 71# Initialize some time variables used in module functions. 72start_time = time.time() 73# sprint_time_last_seconds is used to calculate elapsed seconds. 74sprint_time_last_seconds = [start_time, start_time] 75# Define global index for the sprint_time_last_seconds list. 76last_seconds_ix = 0 77 78 79def set_last_seconds_ix(ix): 80 r""" 81 Set the "last_seconds_ix" module variable to the index value. 82 83 Description of argument(s): 84 ix The index value to be set into the module global last_seconds_ix variable. 85 """ 86 global last_seconds_ix 87 last_seconds_ix = ix 88 89 90# Since output from the lprint_ functions goes to a different location than the output from the print_ 91# functions (e.g. a file vs. the console), sprint_time_last_seconds has been created as a list rather than a 92# simple integer so that it can store multiple sprint_time_last_seconds values. Standard print_ functions 93# defined in this file will use sprint_time_last_seconds[0] and the lprint_ functions will use 94# sprint_time_last_seconds[1]. 95def standard_print_last_seconds_ix(): 96 r""" 97 Return the standard print last_seconds index value to the caller. 98 """ 99 return 0 100 101 102def lprint_last_seconds_ix(): 103 r""" 104 Return lprint last_seconds index value to the caller. 105 """ 106 return 1 107 108 109# The user can set environment variable "GEN_PRINT_DEBUG" to get debug output from this module. 110gen_print_debug = int(os.environ.get('GEN_PRINT_DEBUG', 0)) 111 112 113def sprint_func_name(stack_frame_ix=None): 114 r""" 115 Return the function name associated with the indicated stack frame. 116 117 Description of argument(s): 118 stack_frame_ix The index of the stack frame whose function name should be returned. If 119 the caller does not specify a value, this function will set the value to 120 1 which is the index of the caller's stack frame. If the caller is the 121 wrapper function "print_func_name", this function will bump it up by 1. 122 """ 123 124 # If user specified no stack_frame_ix, we'll set it to a proper default value. 125 if stack_frame_ix is None: 126 func_name = sys._getframe().f_code.co_name 127 caller_func_name = sys._getframe(1).f_code.co_name 128 if func_name[1:] == caller_func_name: 129 stack_frame_ix = 2 130 else: 131 stack_frame_ix = 1 132 133 func_name = sys._getframe(stack_frame_ix).f_code.co_name 134 135 return func_name 136 137 138def work_around_inspect_stack_cwd_failure(): 139 r""" 140 Work around the inspect.stack() getcwd() failure by making "/tmp" the current working directory. 141 142 NOTES: If the current working directory has been deleted, inspect.stack() will fail with "OSError: [Errno 143 2] No such file or directory" because it tries to do a getcwd(). 144 145 This function will try to prevent this failure by detecting the scenario in advance and making "/tmp" the 146 current working directory. 147 """ 148 try: 149 os.getcwd() 150 except OSError: 151 os.chdir("/tmp") 152 153 154def get_line_indent(line): 155 r""" 156 Return the number of spaces at the beginning of the line. 157 """ 158 159 return len(line) - len(line.lstrip(' ')) 160 161 162# get_arg_name is not a print function per se. It has been included in this module because it is used by 163# sprint_var which is defined in this module. 164def get_arg_name(var, 165 arg_num=1, 166 stack_frame_ix=1): 167 r""" 168 Return the "name" of an argument passed to a function. This could be a literal or a variable name. 169 170 Description of argument(s): 171 var The variable whose name is to be returned. 172 arg_num The arg number whose name is to be returned. To illustrate how arg_num 173 is processed, suppose that a programmer codes this line: "rc, outbuf = 174 my_func(var1, var2)" and suppose that my_func has this line of code: 175 "result = gp.get_arg_name(0, arg_num, 2)". If arg_num is positive, the 176 indicated argument is returned. For example, if arg_num is 1, "var1" 177 would be returned, If arg_num is 2, "var2" would be returned. If arg_num 178 exceeds the number of arguments, get_arg_name will simply return a 179 complete list of the arguments. If arg_num is 0, get_arg_name will 180 return the name of the target function as specified in the calling line 181 ("my_func" in this case). To clarify, if the caller of the target 182 function uses an alias function name, the alias name would be returned. 183 If arg_num is negative, an lvalue variable name is returned. Continuing 184 with the given example, if arg_num is -2 the 2nd parm to the left of the 185 "=" ("rc" in this case) should be returned. If arg_num is -1, the 1st 186 parm to the left of the "=" ("out_buf" in this case) should be returned. 187 If arg_num is less than -2, an entire dictionary is returned. The keys 188 to the dictionary for this example would be -2 and -1. 189 stack_frame_ix The stack frame index of the target function. This value must be 1 or 190 greater. 1 would indicate get_arg_name's stack frame. 2 would be the 191 caller of get_arg_name's stack frame, etc. 192 193 Example 1: 194 195 my_var = "mike" 196 var_name = get_arg_name(my_var) 197 198 In this example, var_name will receive the value "my_var". 199 200 Example 2: 201 202 def test1(var): 203 # Getting the var name of the first arg to this function, test1. Note, in this case, it doesn't 204 # matter what is passed as the first arg to get_arg_name since it is the caller's variable name that 205 # matters. 206 dummy = 1 207 arg_num = 1 208 stack_frame = 2 209 var_name = get_arg_name(dummy, arg_num, stack_frame) 210 211 # Mainline... 212 213 another_var = "whatever" 214 test1(another_var) 215 216 In this example, var_name will be set to "another_var". 217 218 """ 219 220 # Note: To avoid infinite recursion, avoid calling any function that calls this function (e.g. 221 # sprint_var, valid_value, etc.). 222 223 # The user can set environment variable "GET_ARG_NAME_DEBUG" to get debug output from this function. 224 local_debug = int(os.environ.get('GET_ARG_NAME_DEBUG', 0)) 225 # In addition to GET_ARG_NAME_DEBUG, the user can set environment variable "GET_ARG_NAME_SHOW_SOURCE" to 226 # have this function include source code in the debug output. 227 local_debug_show_source = int( 228 os.environ.get('GET_ARG_NAME_SHOW_SOURCE', 0)) 229 230 if stack_frame_ix < 1: 231 print_error("Programmer error - Variable \"stack_frame_ix\" has an" 232 + " invalid value of \"" + str(stack_frame_ix) + "\". The" 233 + " value must be an integer that is greater than or equal" 234 + " to 1.\n") 235 return 236 237 if local_debug: 238 debug_indent = 2 239 print("") 240 print_dashes(0, 120) 241 print(sprint_func_name() + "() parms:") 242 print_varx("var", var, indent=debug_indent) 243 print_varx("arg_num", arg_num, indent=debug_indent) 244 print_varx("stack_frame_ix", stack_frame_ix, indent=debug_indent) 245 print("") 246 print_call_stack(debug_indent, 2) 247 248 work_around_inspect_stack_cwd_failure() 249 for count in range(0, 2): 250 try: 251 frame, filename, cur_line_no, function_name, lines, index = \ 252 inspect.stack()[stack_frame_ix] 253 except IndexError: 254 print_error("Programmer error - The caller has asked for" 255 + " information about the stack frame at index \"" 256 + str(stack_frame_ix) + "\". However, the stack" 257 + " only contains " + str(len(inspect.stack())) 258 + " entries. Therefore the stack frame index is out" 259 + " of range.\n") 260 return 261 if filename != "<string>": 262 break 263 # filename of "<string>" may mean that the function in question was defined dynamically and 264 # therefore its code stack is inaccessible. This may happen with functions like "rqprint_var". In 265 # this case, we'll increment the stack_frame_ix and try again. 266 stack_frame_ix += 1 267 if local_debug: 268 print("Adjusted stack_frame_ix...") 269 print_varx("stack_frame_ix", stack_frame_ix, indent=debug_indent) 270 271 real_called_func_name = sprint_func_name(stack_frame_ix) 272 273 module = inspect.getmodule(frame) 274 275 # Though one would expect inspect.getsourcelines(frame) to get all module source lines if the frame is 276 # "<module>", it doesn't do that. Therefore, for this special case, do inspect.getsourcelines(module). 277 if function_name == "<module>": 278 source_lines, source_line_num =\ 279 inspect.getsourcelines(module) 280 line_ix = cur_line_no - source_line_num - 1 281 else: 282 source_lines, source_line_num =\ 283 inspect.getsourcelines(frame) 284 line_ix = cur_line_no - source_line_num 285 286 if local_debug: 287 print("\n Variables retrieved from inspect.stack() function:") 288 print_varx("frame", frame, indent=debug_indent + 2) 289 print_varx("filename", filename, indent=debug_indent + 2) 290 print_varx("cur_line_no", cur_line_no, indent=debug_indent + 2) 291 print_varx("function_name", function_name, indent=debug_indent + 2) 292 print_varx("lines", lines, indent=debug_indent + 2) 293 print_varx("index", index, indent=debug_indent + 2) 294 print_varx("source_line_num", source_line_num, indent=debug_indent) 295 print_varx("line_ix", line_ix, indent=debug_indent) 296 if local_debug_show_source: 297 print_varx("source_lines", source_lines, indent=debug_indent) 298 print_varx("real_called_func_name", real_called_func_name, 299 indent=debug_indent) 300 301 # Get a list of all functions defined for the module. Note that this doesn't work consistently when 302 # _run_exitfuncs is at the top of the stack (i.e. if we're running an exit function). I've coded a 303 # work-around below for this deficiency. 304 all_functions = inspect.getmembers(module, inspect.isfunction) 305 306 # Get called_func_id by searching for our function in the list of all functions. 307 called_func_id = None 308 for func_name, function in all_functions: 309 if func_name == real_called_func_name: 310 called_func_id = id(function) 311 break 312 # NOTE: The only time I've found that called_func_id can't be found is when we're running from an exit 313 # function. 314 315 # Look for other functions in module with matching id. 316 aliases = set([real_called_func_name]) 317 for func_name, function in all_functions: 318 if func_name == real_called_func_name: 319 continue 320 func_id = id(function) 321 if func_id == called_func_id: 322 aliases.add(func_name) 323 324 # In most cases, my general purpose code above will find all aliases. However, for the odd case (i.e. 325 # running from exit function), I've added code to handle pvar, qpvar, dpvar, etc. aliases explicitly 326 # since they are defined in this module and used frequently. 327 # pvar is an alias for print_var. 328 aliases.add(re.sub("print_var", "pvar", real_called_func_name)) 329 330 # The call to the function could be encased in a recast (e.g. int(func_name())). 331 recast_regex = "([^ ]+\\([ ]*)?" 332 import_name_regex = "([a-zA-Z0-9_]+\\.)?" 333 func_name_regex = recast_regex + import_name_regex + "(" +\ 334 '|'.join(aliases) + ")" 335 pre_args_regex = ".*" + func_name_regex + "[ ]*\\(" 336 337 # Search backward through source lines looking for the calling function name. 338 found = False 339 for start_line_ix in range(line_ix, 0, -1): 340 # Skip comment lines. 341 if re.match(r"[ ]*#", source_lines[start_line_ix]): 342 continue 343 if re.match(pre_args_regex, source_lines[start_line_ix]): 344 found = True 345 break 346 if not found: 347 print_error("Programmer error - Could not find the source line with" 348 + " a reference to function \"" + real_called_func_name 349 + "\".\n") 350 return 351 352 # Search forward through the source lines looking for a line whose indentation is the same or less than 353 # the start line. The end of our composite line should be the line preceding that line. 354 start_indent = get_line_indent(source_lines[start_line_ix]) 355 end_line_ix = line_ix 356 for end_line_ix in range(line_ix + 1, len(source_lines)): 357 if source_lines[end_line_ix].strip() == "": 358 continue 359 line_indent = get_line_indent(source_lines[end_line_ix]) 360 if line_indent <= start_indent: 361 end_line_ix -= 1 362 break 363 if start_line_ix != 0: 364 # Check to see whether the start line is a continuation of the prior line. 365 prior_line = source_lines[start_line_ix - 1] 366 prior_line_stripped = re.sub(r"[ ]*\\([\r\n]$)", " \\1", prior_line) 367 prior_line_indent = get_line_indent(prior_line) 368 if prior_line != prior_line_stripped and\ 369 prior_line_indent < start_indent: 370 start_line_ix -= 1 371 # Remove the backslash (continuation char) from prior line. 372 source_lines[start_line_ix] = prior_line_stripped 373 374 # Join the start line through the end line into a composite line. 375 composite_line = ''.join(map(str.strip, 376 source_lines[start_line_ix:end_line_ix + 1])) 377 # Insert one space after first "=" if there isn't one already. 378 composite_line = re.sub("=[ ]*([^ ])", "= \\1", composite_line, 1) 379 380 lvalue_regex = "[ ]*=[ ]+" + func_name_regex + ".*" 381 lvalue_string = re.sub(lvalue_regex, "", composite_line) 382 if lvalue_string == composite_line: 383 # i.e. the regex did not match so there are no lvalues. 384 lvalue_string = "" 385 lvalues_list = list(filter(None, map(str.strip, lvalue_string.split(",")))) 386 try: 387 lvalues = collections.OrderedDict() 388 except AttributeError: 389 # A non-ordered dict doesn't look as nice when printed but it will do. 390 lvalues = {} 391 ix = len(lvalues_list) * -1 392 for lvalue in lvalues_list: 393 lvalues[ix] = lvalue 394 ix += 1 395 lvalue_prefix_regex = "(.*=[ ]+)?" 396 called_func_name_regex = lvalue_prefix_regex + func_name_regex +\ 397 "[ ]*\\(.*" 398 called_func_name = re.sub(called_func_name_regex, "\\4", composite_line) 399 arg_list_etc = "(" + re.sub(pre_args_regex, "", composite_line) 400 if local_debug: 401 print_varx("aliases", aliases, indent=debug_indent) 402 print_varx("import_name_regex", import_name_regex, indent=debug_indent) 403 print_varx("func_name_regex", func_name_regex, indent=debug_indent) 404 print_varx("pre_args_regex", pre_args_regex, indent=debug_indent) 405 print_varx("start_line_ix", start_line_ix, indent=debug_indent) 406 print_varx("end_line_ix", end_line_ix, indent=debug_indent) 407 print_varx("composite_line", composite_line, indent=debug_indent) 408 print_varx("lvalue_regex", lvalue_regex, indent=debug_indent) 409 print_varx("lvalue_string", lvalue_string, indent=debug_indent) 410 print_varx("lvalues", lvalues, indent=debug_indent) 411 print_varx("called_func_name_regex", called_func_name_regex, 412 indent=debug_indent) 413 print_varx("called_func_name", called_func_name, indent=debug_indent) 414 print_varx("arg_list_etc", arg_list_etc, indent=debug_indent) 415 416 # Parse arg list... 417 # Initialize... 418 nest_level = -1 419 arg_ix = 0 420 args_list = [""] 421 for ix in range(0, len(arg_list_etc)): 422 char = arg_list_etc[ix] 423 # Set the nest_level based on whether we've encounted a parenthesis. 424 if char == "(": 425 nest_level += 1 426 if nest_level == 0: 427 continue 428 elif char == ")": 429 nest_level -= 1 430 if nest_level < 0: 431 break 432 433 # If we reach a comma at base nest level, we are done processing an argument so we increment arg_ix 434 # and initialize a new args_list entry. 435 if char == "," and nest_level == 0: 436 arg_ix += 1 437 args_list.append("") 438 continue 439 440 # For any other character, we append it it to the current arg list entry. 441 args_list[arg_ix] += char 442 443 # Trim whitespace from each list entry. 444 args_list = [arg.strip() for arg in args_list] 445 446 if arg_num < 0: 447 if abs(arg_num) > len(lvalues): 448 argument = lvalues 449 else: 450 argument = lvalues[arg_num] 451 elif arg_num == 0: 452 argument = called_func_name 453 else: 454 if arg_num > len(args_list): 455 argument = args_list 456 else: 457 argument = args_list[arg_num - 1] 458 459 if local_debug: 460 print_varx("args_list", args_list, indent=debug_indent) 461 print_varx("argument", argument, indent=debug_indent) 462 print_dashes(0, 120) 463 464 return argument 465 466 467def sprint_time(buffer=""): 468 r""" 469 Return the time in the following format. 470 471 Example: 472 473 The following python code... 474 475 sys.stdout.write(sprint_time()) 476 sys.stdout.write("Hi.\n") 477 478 Will result in the following type of output: 479 480 #(CDT) 2016/07/08 15:25:35 - Hi. 481 482 Example: 483 484 The following python code... 485 486 sys.stdout.write(sprint_time("Hi.\n")) 487 488 Will result in the following type of output: 489 490 #(CDT) 2016/08/03 17:12:05 - Hi. 491 492 The following environment variables will affect the formatting as described: 493 NANOSECONDS This will cause the time stamps to be precise to the microsecond (Yes, it 494 probably should have been named MICROSECONDS but the convention was set 495 long ago so we're sticking with it). Example of the output when 496 environment variable NANOSECONDS=1. 497 498 #(CDT) 2016/08/03 17:16:25.510469 - Hi. 499 500 SHOW_ELAPSED_TIME This will cause the elapsed time to be included in the output. This is 501 the amount of time that has elapsed since the last time this function was 502 called. The precision of the elapsed time field is also affected by the 503 value of the NANOSECONDS environment variable. Example of the output 504 when environment variable NANOSECONDS=0 and SHOW_ELAPSED_TIME=1. 505 506 #(CDT) 2016/08/03 17:17:40 - 0 - Hi. 507 508 Example of the output when environment variable NANOSECONDS=1 and SHOW_ELAPSED_TIME=1. 509 510 #(CDT) 2016/08/03 17:18:47.317339 - 0.000046 - Hi. 511 512 Description of argument(s). 513 buffer This will be appended to the formatted time string. 514 """ 515 516 global NANOSECONDS 517 global SHOW_ELAPSED_TIME 518 global sprint_time_last_seconds 519 global last_seconds_ix 520 521 seconds = time.time() 522 loc_time = time.localtime(seconds) 523 nanoseconds = "%0.6f" % seconds 524 pos = nanoseconds.find(".") 525 nanoseconds = nanoseconds[pos:] 526 527 time_string = time.strftime("#(%Z) %Y/%m/%d %H:%M:%S", loc_time) 528 if NANOSECONDS == "1": 529 time_string = time_string + nanoseconds 530 531 if SHOW_ELAPSED_TIME == "1": 532 cur_time_seconds = seconds 533 math_string = "%9.9f" % cur_time_seconds + " - " + "%9.9f" % \ 534 sprint_time_last_seconds[last_seconds_ix] 535 elapsed_seconds = eval(math_string) 536 if NANOSECONDS == "1": 537 elapsed_seconds = "%11.6f" % elapsed_seconds 538 else: 539 elapsed_seconds = "%4i" % elapsed_seconds 540 sprint_time_last_seconds[last_seconds_ix] = cur_time_seconds 541 time_string = time_string + " - " + elapsed_seconds 542 543 return time_string + " - " + buffer 544 545 546def sprint_timen(buffer=""): 547 r""" 548 Append a line feed to the buffer, pass it to sprint_time and return the result. 549 """ 550 551 return sprint_time(buffer + "\n") 552 553 554def sprint_error(buffer=""): 555 r""" 556 Return a standardized error string. This includes: 557 - A time stamp 558 - The "**ERROR**" string 559 - The caller's buffer string. 560 561 Example: 562 563 The following python code... 564 565 print(sprint_error("Oops.\n")) 566 567 Will result in the following type of output: 568 569 #(CDT) 2016/08/03 17:12:05 - **ERROR** Oops. 570 571 Description of argument(s). 572 buffer This will be appended to the formatted error string. 573 """ 574 575 return sprint_time() + "**ERROR** " + buffer 576 577 578# Implement "constants" with functions. 579def digit_length_in_bits(): 580 r""" 581 Return the digit length in bits. 582 """ 583 584 return 4 585 586 587def word_length_in_digits(): 588 r""" 589 Return the word length in digits. 590 """ 591 592 return 8 593 594 595def bit_length(number): 596 r""" 597 Return the bit length of the number. 598 599 Description of argument(s): 600 number The number to be analyzed. 601 """ 602 603 if number < 0: 604 # Convert negative numbers to positive and subtract one. The following example illustrates the 605 # reason for this: 606 # Consider a single nibble whose signed values can range from -8 to 7 (0x8 to 0x7). A value of 0x7 607 # equals 0b0111. Therefore, its length in bits is 3. Since the negative bit (i.e. 0b1000) is not 608 # set, the value 7 clearly will fit in one nibble. With -8 = 0x8 = 0b1000, one has the smallest 609 # negative value that will fit. Note that it requires 3 bits of 0. So by converting a number value 610 # of -8 to a working_number of 7, this function can accurately calculate the number of bits and 611 # therefore nibbles required to represent the number in print. 612 working_number = abs(number) - 1 613 else: 614 working_number = number 615 616 # Handle the special case of the number 0. 617 if working_number == 0: 618 return 0 619 620 return len(bin(working_number)) - 2 621 622 623def get_req_num_hex_digits(number): 624 r""" 625 Return the required number of hex digits required to display the given number. 626 627 The returned value will always be rounded up to the nearest multiple of 8. 628 629 Description of argument(s): 630 number The number to be analyzed. 631 """ 632 633 if number < 0: 634 # Convert negative numbers to positive and subtract one. The following example illustrates the 635 # reason for this: 636 # Consider a single nibble whose signed values can range from -8 to 7 (0x8 to 0x7). A value of 0x7 637 # equals 0b0111. Therefore, its length in bits is 3. Since the negative bit (i.e. 0b1000) is not 638 # set, the value 7 clearly will fit in one nibble. With -8 = 0x8 = 0b1000, one has the smallest 639 # negative value that will fit. Note that it requires 3 bits of 0. So by converting a number value 640 # of -8 to a working_number of 7, this function can accurately calculate the number of bits and 641 # therefore nibbles required to represent the number in print. 642 working_number = abs(number) - 1 643 else: 644 working_number = number 645 646 # Handle the special case of the number 0. 647 if working_number == 0: 648 return word_length_in_digits() 649 650 num_length_in_bits = bit_length(working_number) 651 num_hex_digits, remainder = divmod(num_length_in_bits, 652 digit_length_in_bits()) 653 if remainder > 0: 654 # Example: the number 7 requires 3 bits. The divmod above produces, 0 with remainder of 3. So 655 # because we have a remainder, we increment num_hex_digits from 0 to 1. 656 num_hex_digits += 1 657 658 # Check to see whether the negative bit is set. This is the left-most bit in the highest order digit. 659 negative_mask = 2 ** (num_hex_digits * 4 - 1) 660 if working_number & negative_mask: 661 # If a number that is intended to be positive has its negative bit on, an additional digit will be 662 # required to represent it correctly in print. 663 num_hex_digits += 1 664 665 num_words, remainder = divmod(num_hex_digits, word_length_in_digits()) 666 if remainder > 0 or num_words == 0: 667 num_words += 1 668 669 # Round up to the next word length in digits. 670 return num_words * word_length_in_digits() 671 672 673def dft_num_hex_digits(): 674 r""" 675 Return the default number of hex digits to be used to represent a hex number in print. 676 677 The value returned is a function of sys.maxsize. 678 """ 679 680 global _gen_print_dft_num_hex_digits_ 681 try: 682 return _gen_print_dft_num_hex_digits_ 683 except NameError: 684 _gen_print_dft_num_hex_digits_ = get_req_num_hex_digits(sys.maxsize) 685 return _gen_print_dft_num_hex_digits_ 686 687 688# Create constant functions to describe various types of dictionaries. 689def dict_type(): 690 return 1 691 692 693def ordered_dict_type(): 694 return 2 695 696 697def dot_dict_type(): 698 return 3 699 700 701def normalized_dict_type(): 702 return 4 703 704 705def proxy_dict_type(): 706 return 5 707 708 709def is_dict(var_value): 710 r""" 711 Return non-zero if var_value is a type of dictionary and 0 if it is not. 712 713 The specific non-zero value returned will indicate what type of dictionary var_value is (see constant 714 functions above). 715 716 Description of argument(s): 717 var_value The object to be analyzed to determine whether it is a dictionary and if 718 so, what type of dictionary. 719 """ 720 721 if isinstance(var_value, dict): 722 return dict_type() 723 try: 724 if isinstance(var_value, collections.OrderedDict): 725 return ordered_dict_type() 726 except AttributeError: 727 pass 728 try: 729 if isinstance(var_value, DotDict): 730 return dot_dict_type() 731 except NameError: 732 pass 733 try: 734 if isinstance(var_value, NormalizedDict): 735 return normalized_dict_type() 736 except NameError: 737 pass 738 try: 739 if str(type(var_value)).split("'")[1] == "dictproxy": 740 return proxy_dict_type() 741 except NameError: 742 pass 743 return 0 744 745 746def get_int_types(): 747 r""" 748 Return a tuple consisting of the valid integer data types for the system and version of python being run. 749 750 Example: 751 (int, long) 752 """ 753 754 try: 755 int_types = (int, long) 756 except NameError: 757 int_types = (int,) 758 return int_types 759 760 761def get_string_types(): 762 r""" 763 Return a tuple consisting of the valid string data types for the system and version of python being run. 764 765 Example: 766 (str, unicode) 767 """ 768 769 try: 770 string_types = (str, unicode) 771 except NameError: 772 string_types = (bytes, str) 773 return string_types 774 775 776def valid_fmts(): 777 r""" 778 Return a list of the valid formats that can be specified for the fmt argument of the sprint_varx function 779 (defined below). 780 """ 781 782 return [ 783 'hexa', 784 'octal', 785 'binary', 786 'blank', 787 'verbose', 788 'quote_keys', 789 'show_type', 790 'strip_brackets', 791 'no_header', 792 'quote_values'] 793 794 795def create_fmt_definition(): 796 r""" 797 Create a string consisting of function-definition code that can be executed to create constant fmt 798 definition functions. 799 800 These functions can be used by callers of sprint_var/sprint_varx to set the fmt argument correctly. 801 802 Likewise, the sprint_varx function will use these generated functions to correctly interpret the fmt 803 argument. 804 805 Example output from this function: 806 807 def hexa(): 808 return 0x00000001 809 def octal_fmt(): 810 return 0x00000002 811 etc. 812 """ 813 814 buffer = "" 815 bits = 0x00000001 816 for fmt_name in valid_fmts(): 817 buffer += "def " + fmt_name + "():\n" 818 buffer += " return " + "0x%08x" % bits + "\n" 819 bits = bits << 1 820 return buffer 821 822 823# Dynamically create fmt definitions (for use with the fmt argument of sprint_varx function): 824exec(create_fmt_definition()) 825 826 827def terse(): 828 r""" 829 Constant function to return fmt value of 0. 830 831 Now that sprint_varx defaults to printing in terse format, the terse option is deprecated. This function 832 is here for backward compatibility. 833 834 Once the repo has been purged of the use of terse, this function can be removed. 835 """ 836 837 return 0 838 839 840def list_pop(a_list, index=0, default=None): 841 r""" 842 Pop the list entry indicated by the index and return the entry. If no such entry exists, return default. 843 844 Note that the list passed to this function will be modified. 845 846 Description of argument(s): 847 a_list The list from which an entry is to be popped. 848 index The index indicating which entry is to be popped. 849 default The value to be returned if there is no entry at the given index location. 850 """ 851 try: 852 return a_list.pop(index) 853 except IndexError: 854 return default 855 856 857def parse_fmt(fmt): 858 r""" 859 Parse the fmt argument and return a tuple consisting of a format and a child format. 860 861 This function was written for use by the sprint_varx function defined in this module. 862 863 When sprint_varx is processing a multi-level object such as a list or dictionary (which in turn may 864 contain other lists or dictionaries), it will use the fmt value to dictate the print formatting of the 865 current level and the child_fmt value to dictate the print formatting of subordinate levels. Consider 866 the following example: 867 868 python code example: 869 870 ord_dict = \ 871 collections.OrderedDict([ 872 ('one', 1), 873 ('two', 2), 874 ('sub', 875 collections.OrderedDict([ 876 ('three', 3), ('four', 4)]))]) 877 878 print_var(ord_dict) 879 880 This would generate the following output: 881 882 ord_dict: 883 [one]: 1 884 [two]: 2 885 [sub]: 886 [three]: 3 887 [four]: 4 888 889 The first level in this example is the line that simply says "ord_dict". The second level is comprised 890 of the dictionary entries with the keys 'one', 'two' and 'sub'. The third level is comprised of the last 891 2 lines (i.e. printed values 3 and 4). 892 893 Given the data structure shown above, the programmer could code the following where fmt is a simple 894 integer value set by calling the verbose() function. 895 896 print_var(ord_dict, fmt=verbose()) 897 898 The output would look like this: 899 900 ord_dict: 901 ord_dict[one]: 1 902 ord_dict[two]: 2 903 ord_dict[sub]: 904 ord_dict[sub][three]: 3 905 ord_dict[sub][four]: 4 906 907 Note the verbose format where the name of the object ("ord_dict") is repeated on every line. 908 909 If the programmer wishes to get more granular with the fmt argument, he/she can specify it as a list 910 where each entry corresponds to a level of the object being printed. The last such list entry governs 911 the print formatting of all subordinate parts of the given object. 912 913 Look at each of the following code examples and their corresponding output. See how the show_type() 914 formatting affects the printing depending on which position it occupies in the fmt list argument: 915 916 print_var(ord_dict, fmt=[show_type()]) 917 918 ord_dict: <collections.OrderedDict> 919 ord_dict[one]: 1 <int> 920 ord_dict[two]: 2 <int> 921 ord_dict[sub]: <collections.OrderedDict> 922 ord_dict[sub][three]: 3 <int> 923 ord_dict[sub][four]: 4 <int> 924 925 print_var(ord_dict, fmt=[0, show_type()]) 926 927 ord_dict: 928 ord_dict[one]: 1 <int> 929 ord_dict[two]: 2 <int> 930 ord_dict[sub]: <collections.OrderedDict> 931 ord_dict[sub][three]: 3 <int> 932 ord_dict[sub][four]: 4 <int> 933 934 print_var(ord_dict, fmt=[0, 0, show_type()]) 935 936 ord_dict: 937 ord_dict[one]: 1 938 ord_dict[two]: 2 939 ord_dict[sub]: 940 ord_dict[sub][three]: 3 <int> 941 ord_dict[sub][four]: 4 <int> 942 943 Description of argument(s): 944 fmt The format argument such as is passed to sprint_varx. This argument may 945 be an integer or a list of integers. See the prolog of sprint_varx for 946 more details. 947 """ 948 949 # Make a deep copy of the fmt argument in order to avoid modifying the caller's fmt value when it is a 950 # list. 951 fmt = copy.deepcopy(fmt) 952 try: 953 # Assume fmt is a list. Pop the first element from the list. 954 first_element = list_pop(fmt, index=0, default=0) 955 # Return the first list element along with either 1) the remainder of the fmt list if not null or 2) 956 # another copy of the first element. 957 return first_element, fmt if len(fmt) else first_element 958 except AttributeError: 959 # fmt is not a list so treat it as a simple integer value. 960 return fmt, fmt 961 962 963def sprint_varx(var_name, 964 var_value, 965 fmt=0, 966 indent=dft_indent, 967 col1_width=dft_col1_width, 968 trailing_char="\n", 969 key_list=None, 970 delim=":"): 971 r""" 972 Print the var name/value passed to it. If the caller lets col1_width default, the printing lines up 973 nicely with output generated by the print_time functions. 974 975 Note that the sprint_var function (defined below) can be used to call this function so that the 976 programmer does not need to pass the var_name. sprint_var will figure out the var_name. The sprint_var 977 function is the one that would normally be used by the general user. 978 979 For example, the following python code: 980 981 first_name = "Mike" 982 print_time("Doing this...\n") 983 print_varx("first_name", first_name) 984 print_time("Doing that...\n") 985 986 Will generate output like this: 987 988 #(CDT) 2016/08/10 17:34:42.847374 - 0.001285 - Doing this... 989 first_name: Mike 990 #(CDT) 2016/08/10 17:34:42.847510 - 0.000136 - Doing that... 991 992 This function recognizes several complex types of data such as dict, list or tuple. 993 994 For example, the following python code: 995 996 my_dict = dict(one=1, two=2, three=3) 997 print_var(my_dict) 998 999 Will generate the following output: 1000 1001 my_dict: 1002 my_dict[three]: 3 1003 my_dict[two]: 2 1004 my_dict[one]: 1 1005 1006 Description of argument(s). 1007 var_name The name of the variable to be printed. 1008 var_value The value of the variable to be printed. 1009 fmt A bit map to dictate the format of the output. For printing multi-level 1010 objects like lists and dictionaries, this argument may also be a list of 1011 bit maps. The first list element pertains to the highest level of 1012 output, the second element pertains to the 2nd level of output, etc. The 1013 last element in the list pertains to all subordinate levels. The bits 1014 can be set using the dynamically created functionhs above. Example: 1015 sprint_varx("var1", var1, fmt=verbose()). Note that these values can be 1016 OR'ed together: print_var(var1, hexa() | verbose()). If the caller ORs 1017 mutually exclusive bits (hexa() | octal()), behavior is not guaranteed. 1018 The following features are supported: 1019 hexa Print all integer values in hexadecimal format. 1020 octal Print all integer values in octal format. 1021 binary Print all integer values in binary format. 1022 blank For blank string values, print "<blank>" instead of an actual blank. 1023 verbose For structured values like dictionaries, lists, etc. repeat the name of 1024 the variable on each line to the right of the key or subscript value. 1025 Example: print "my_dict[key1]" instead of just "[key1]". 1026 quote_keys Quote dictionary keys in the output. Example: my_dict['key1'] instead of 1027 my_dict[key1]. 1028 show_type Show the type of the data in angled brackets just to the right of the 1029 data. 1030 strip_brackets Strip the brackets from the variable name portion of the output. This is 1031 applicable when printing complex objects like lists or dictionaries. 1032 no_header For complex objects like dictionaries, do not include a header line. 1033 This necessarily means that the member lines will be indented 2 1034 characters less than they otherwise would have been. 1035 quote_values Quote the values printed. 1036 indent The number of spaces to indent the output. 1037 col1_width The width of the output column containing the variable name. The default 1038 value of this is adjusted so that the var_value lines up with text 1039 printed via the print_time function. 1040 trailing_char The character to be used at the end of the returned string. The default 1041 value is a line feed. 1042 key_list A list of which dictionary keys should be printed. All others keys will 1043 be skipped. Each value in key_list will be regarded as a regular 1044 expression and it will be regarded as anchored to the beginning and ends 1045 of the dictionary key being referenced. For example if key_list is 1046 ["one", "two"], the resulting regex used will be "^one|two$", i.e. only 1047 keys "one" and "two" from the var_value dictionary will be printed. As 1048 another example, if the caller were to specify a key_list of ["one.*"], 1049 then only dictionary keys whose names begin with "one" will be printed. 1050 Note: This argument pertains only to var_values which are dictionaries. 1051 delim The value to be used to delimit the variable name from the variable value 1052 in the output. 1053 """ 1054 1055 fmt, child_fmt = parse_fmt(fmt) 1056 1057 if fmt & show_type(): 1058 type_str = "<" + str(type(var_value)).split("'")[1] + ">" 1059 # Compose object type categories. 1060 int_types = get_int_types() 1061 string_types = get_string_types() 1062 simple_types = int_types + string_types + (float, bool, type, type(None)) 1063 # Determine the type. 1064 if type(var_value) in simple_types: 1065 # The data type is simple in the sense that it has no subordinate parts. 1066 # Adjust col1_width. 1067 col1_width = col1_width - indent 1068 # Set default value for value_format. 1069 value_format = "%s" 1070 # Process format requests. 1071 if type(var_value) in int_types: 1072 # Process format values pertaining to int types. 1073 if fmt & hexa(): 1074 num_hex_digits = max(dft_num_hex_digits(), 1075 get_req_num_hex_digits(var_value)) 1076 # Convert a negative number to its positive twos complement for proper printing. For 1077 # example, instead of printing -1 as "0x-000000000000001" it will be printed as 1078 # "0xffffffffffffffff". 1079 var_value = var_value & (2 ** (num_hex_digits * 4) - 1) 1080 value_format = "0x%0" + str(num_hex_digits) + "x" 1081 elif fmt & octal(): 1082 value_format = "0o%016o" 1083 elif fmt & binary(): 1084 num_digits, remainder = \ 1085 divmod(max(bit_length(var_value), 1), 8) 1086 num_digits *= 8 1087 if remainder: 1088 num_digits += 8 1089 num_digits += 2 1090 value_format = '#0' + str(num_digits) + 'b' 1091 var_value = format(var_value, value_format) 1092 value_format = "%s" 1093 elif type(var_value) in string_types: 1094 # Process format values pertaining to string types. 1095 if fmt & blank() and var_value == "": 1096 value_format = "%s" 1097 var_value = "<blank>" 1098 elif type(var_value) is type: 1099 var_value = str(var_value).split("'")[1] 1100 format_string = "%" + str(indent) + "s%-" + str(col1_width) + "s" \ 1101 + value_format 1102 if fmt & show_type(): 1103 if var_value != "": 1104 format_string += " " 1105 format_string += type_str 1106 format_string += trailing_char 1107 if fmt & quote_values(): 1108 var_value = "'" + var_value + "'" 1109 if not (fmt & verbose()): 1110 # Strip everything leading up to the first left square brace. 1111 var_name = re.sub(r".*\[", "[", var_name) 1112 if (fmt & strip_brackets()): 1113 var_name = re.sub(r"[\[\]]", "", var_name) 1114 if value_format == "0x%08x": 1115 return format_string % ("", str(var_name) + delim, 1116 var_value & 0xffffffff) 1117 else: 1118 return format_string % ("", str(var_name) + delim, var_value) 1119 else: 1120 # The data type is complex in the sense that it has subordinate parts. 1121 if (fmt & no_header()): 1122 buffer = "" 1123 else: 1124 # Create header line. 1125 if not (fmt & verbose()): 1126 # Strip everything leading up to the first square brace. 1127 loc_var_name = re.sub(r".*\[", "[", var_name) 1128 else: 1129 loc_var_name = var_name 1130 if (fmt & strip_brackets()): 1131 loc_var_name = re.sub(r"[\[\]]", "", loc_var_name) 1132 format_string = "%" + str(indent) + "s%s\n" 1133 buffer = format_string % ("", loc_var_name + ":") 1134 if fmt & show_type(): 1135 buffer = buffer.replace("\n", " " + type_str + "\n") 1136 indent += 2 1137 try: 1138 length = len(var_value) 1139 except TypeError: 1140 length = 0 1141 ix = 0 1142 loc_trailing_char = "\n" 1143 if is_dict(var_value): 1144 if type(child_fmt) is list: 1145 child_quote_keys = (child_fmt[0] & quote_keys()) 1146 else: 1147 child_quote_keys = (child_fmt & quote_keys()) 1148 for key, value in var_value.items(): 1149 if key_list is not None: 1150 key_list_regex = "^" + "|".join(key_list) + "$" 1151 if not re.match(key_list_regex, key): 1152 continue 1153 ix += 1 1154 if ix == length: 1155 loc_trailing_char = trailing_char 1156 if child_quote_keys: 1157 key = "'" + key + "'" 1158 key = "[" + str(key) + "]" 1159 buffer += sprint_varx(var_name + key, value, child_fmt, indent, 1160 col1_width, loc_trailing_char, key_list, 1161 delim) 1162 elif type(var_value) in (list, tuple, set): 1163 for key, value in enumerate(var_value): 1164 ix += 1 1165 if ix == length: 1166 loc_trailing_char = trailing_char 1167 key = "[" + str(key) + "]" 1168 buffer += sprint_varx(var_name + key, value, child_fmt, indent, 1169 col1_width, loc_trailing_char, key_list, 1170 delim) 1171 elif isinstance(var_value, argparse.Namespace): 1172 for key in var_value.__dict__: 1173 ix += 1 1174 if ix == length: 1175 loc_trailing_char = trailing_char 1176 cmd_buf = "buffer += sprint_varx(var_name + \".\" + str(key)" \ 1177 + ", var_value." + key + ", child_fmt, indent," \ 1178 + " col1_width, loc_trailing_char, key_list," \ 1179 + " delim)" 1180 exec(cmd_buf) 1181 else: 1182 var_type = type(var_value).__name__ 1183 func_name = sys._getframe().f_code.co_name 1184 var_value = "<" + var_type + " type not supported by " + \ 1185 func_name + "()>" 1186 value_format = "%s" 1187 indent -= 2 1188 # Adjust col1_width. 1189 col1_width = col1_width - indent 1190 format_string = "%" + str(indent) + "s%-" \ 1191 + str(col1_width) + "s" + value_format + trailing_char 1192 return format_string % ("", str(var_name) + ":", var_value) 1193 1194 return buffer 1195 1196 return "" 1197 1198 1199def sprint_var(*args, **kwargs): 1200 r""" 1201 Figure out the name of the first argument for the caller and then call sprint_varx with it. Therefore, 1202 the following 2 calls are equivalent: 1203 sprint_varx("var1", var1) 1204 sprint_var(var1) 1205 1206 See sprint_varx for description of arguments. 1207 """ 1208 1209 stack_frame = 2 1210 caller_func_name = sprint_func_name(2) 1211 if caller_func_name.endswith("print_var"): 1212 stack_frame += 1 1213 # Get the name of the first variable passed to this function. 1214 var_name = get_arg_name(None, 1, stack_frame) 1215 return sprint_varx(var_name, *args, **kwargs) 1216 1217 1218def sprint_vars(*args, **kwargs): 1219 r""" 1220 Sprint the values of one or more variables. 1221 1222 Description of argument(s): 1223 args The variable values which are to be printed. 1224 kwargs See sprint_varx (above) for description of additional arguments. 1225 """ 1226 1227 stack_frame = 2 1228 caller_func_name = sprint_func_name(2) 1229 if caller_func_name.endswith("print_vars"): 1230 stack_frame += 1 1231 1232 buffer = "" 1233 arg_num = 1 1234 for var_value in args: 1235 var_name = get_arg_name(None, arg_num, stack_frame) 1236 buffer += sprint_varx(var_name, var_value, **kwargs) 1237 arg_num += 1 1238 1239 return buffer 1240 1241 1242def sprint_dashes(indent=dft_indent, 1243 width=80, 1244 line_feed=1, 1245 char="-"): 1246 r""" 1247 Return a string of dashes to the caller. 1248 1249 Description of argument(s): 1250 indent The number of characters to indent the output. 1251 width The width of the string of dashes. 1252 line_feed Indicates whether the output should end with a line feed. 1253 char The character to be repeated in the output string. 1254 """ 1255 1256 width = int(width) 1257 buffer = " " * int(indent) + char * width 1258 if line_feed: 1259 buffer += "\n" 1260 1261 return buffer 1262 1263 1264def sindent(text="", 1265 indent=0): 1266 r""" 1267 Pre-pend the specified number of characters to the text string (i.e. indent it) and return it. 1268 1269 Description of argument(s): 1270 text The string to be indented. 1271 indent The number of characters to indent the string. 1272 """ 1273 1274 format_string = "%" + str(indent) + "s%s" 1275 buffer = format_string % ("", text) 1276 1277 return buffer 1278 1279 1280func_line_style_std = None 1281func_line_style_short = 1 1282 1283 1284def sprint_func_line(stack_frame, style=None, max_width=160): 1285 r""" 1286 For the given stack_frame, return a formatted string containing the function name and all its arguments. 1287 1288 Example: 1289 1290 func1(last_name = 'walsh', first_name = 'mikey') 1291 1292 Description of argument(s): 1293 stack_frame A stack frame (such as is returned by inspect.stack()). 1294 style Indicates the style or formatting of the result string. Acceptable 1295 values are shown above. 1296 max_width The max width of the result. If it exceeds this length, it will be 1297 truncated on the right. 1298 1299 Description of styles: 1300 func_line_style_std The standard formatting. 1301 func_line_style_short 1) The self parm (associated with methods) will be dropped. 2) The args 1302 and kwargs values will be treated as special. In both cases the arg name 1303 ('args' or 'kwargs') will be dropped and only the values will be shown. 1304 """ 1305 1306 func_name = str(stack_frame[3]) 1307 if func_name == "?": 1308 # "?" is the name used when code is not in a function. 1309 func_name = "(none)" 1310 1311 if func_name == "<module>": 1312 # If the func_name is the "main" program, we simply get the command line call string. 1313 func_and_args = ' '.join(sys.argv) 1314 else: 1315 # Get the program arguments. 1316 (args, varargs, keywords, locals) =\ 1317 inspect.getargvalues(stack_frame[0]) 1318 1319 args_list = [] 1320 for arg_name in filter(None, args + [varargs, keywords]): 1321 # Get the arg value from frame locals. 1322 arg_value = locals[arg_name] 1323 if arg_name == 'self': 1324 if style == func_line_style_short: 1325 continue 1326 # Manipulations to improve output for class methods. 1327 func_name = arg_value.__class__.__name__ + "." + func_name 1328 args_list.append(arg_name + " = <self>") 1329 elif (style == func_line_style_short 1330 and arg_name == 'args' 1331 and type(arg_value) in (list, tuple)): 1332 if len(arg_value) == 0: 1333 continue 1334 args_list.append(repr(', '.join(arg_value))) 1335 elif (style == func_line_style_short 1336 and arg_name == 'kwargs' 1337 and type(arg_value) is dict): 1338 for key, value in arg_value.items(): 1339 args_list.append(key + "=" + repr(value)) 1340 else: 1341 args_list.append(arg_name + " = " + repr(arg_value)) 1342 args_str = "(" + ', '.join(map(str, args_list)) + ")" 1343 1344 # Now we need to print this in a nicely-wrapped way. 1345 func_and_args = func_name + args_str 1346 1347 if len(func_and_args) > max_width: 1348 func_and_args = func_and_args[0:max_width] + "..." 1349 return func_and_args 1350 1351 1352def sprint_call_stack(indent=0, 1353 stack_frame_ix=0, 1354 style=None): 1355 r""" 1356 Return a call stack report for the given point in the program with line numbers, function names and 1357 function parameters and arguments. 1358 1359 Sample output: 1360 1361 ------------------------------------------------------------------------- 1362 Python function call stack 1363 1364 Line # Function name and arguments 1365 ------ ------------------------------------------------------------------ 1366 424 sprint_call_stack() 1367 4 print_call_stack() 1368 31 func1(last_name = 'walsh', first_name = 'mikey') 1369 59 /tmp/scr5.py 1370 ------------------------------------------------------------------------- 1371 1372 Description of argument(s): 1373 indent The number of characters to indent each line of output. 1374 stack_frame_ix The index of the first stack frame which is to be returned. 1375 style See the sprint_line_func prolog above for details. 1376 """ 1377 1378 buffer = "" 1379 buffer += sprint_dashes(indent) 1380 buffer += sindent("Python function call stack\n\n", indent) 1381 buffer += sindent("Line # Function name and arguments\n", indent) 1382 buffer += sprint_dashes(indent, 6, 0) + " " + sprint_dashes(0, 73) 1383 1384 # Grab the current program stack. 1385 work_around_inspect_stack_cwd_failure() 1386 current_stack = inspect.stack() 1387 1388 # Process each frame in turn. 1389 format_string = "%6s %s\n" 1390 ix = 0 1391 for stack_frame in current_stack: 1392 if ix < stack_frame_ix: 1393 ix += 1 1394 continue 1395 # Make the line number shown to be the line where one finds the line shown. 1396 try: 1397 line_num = str(current_stack[ix + 1][2]) 1398 except IndexError: 1399 line_num = "" 1400 func_and_args = sprint_func_line(stack_frame, style=style) 1401 1402 buffer += sindent(format_string % (line_num, func_and_args), indent) 1403 ix += 1 1404 1405 buffer += sprint_dashes(indent) 1406 1407 return buffer 1408 1409 1410def sprint_executing(stack_frame_ix=None, style=None, max_width=None): 1411 r""" 1412 Print a line indicating what function is executing and with what parameter values. This is useful for 1413 debugging. 1414 1415 Sample output: 1416 1417 #(CDT) 2016/08/25 17:54:27 - Executing: func1(x = 1) 1418 1419 Description of argument(s): 1420 stack_frame_ix The index of the stack frame whose function info should be returned. If 1421 the caller does not specify a value, this function will set the value to 1422 1 which is the index of the caller's stack frame. If the caller is the 1423 wrapper function "print_executing", this function will bump it up by 1. 1424 style See the sprint_line_func prolog above for details. 1425 max_width See the sprint_line_func prolog above for details. 1426 """ 1427 1428 # If user wants default stack_frame_ix. 1429 if stack_frame_ix is None: 1430 func_name = sys._getframe().f_code.co_name 1431 caller_func_name = sys._getframe(1).f_code.co_name 1432 if caller_func_name.endswith(func_name[1:]): 1433 stack_frame_ix = 2 1434 else: 1435 stack_frame_ix = 1 1436 1437 work_around_inspect_stack_cwd_failure() 1438 stack_frame = inspect.stack()[stack_frame_ix] 1439 1440 if max_width is None: 1441 max_width = 160 - (dft_col1_width + 11) 1442 func_and_args = sprint_func_line(stack_frame, style, max_width=max_width) 1443 1444 return sprint_time() + "Executing: " + func_and_args + "\n" 1445 1446 1447def sprint_pgm_header(indent=0, 1448 linefeed=1): 1449 r""" 1450 Return a standardized header that programs should print at the beginning of the run. It includes useful 1451 information like command line, pid, userid, program parameters, etc. 1452 1453 Description of argument(s): 1454 indent The number of characters to indent each line of output. 1455 linefeed Indicates whether a line feed be included at the beginning and end of the 1456 report. 1457 """ 1458 1459 col1_width = dft_col1_width + indent 1460 1461 buffer = "" 1462 if linefeed: 1463 buffer = "\n" 1464 1465 if robot_env: 1466 suite_name = BuiltIn().get_variable_value("${suite_name}") 1467 buffer += sindent(sprint_time("Running test suite \"" + suite_name 1468 + "\".\n"), indent) 1469 1470 buffer += sindent(sprint_time() + "Running " + pgm_name + ".\n", indent) 1471 buffer += sindent(sprint_time() + "Program parameter values, etc.:\n\n", 1472 indent) 1473 buffer += sprint_varx("command_line", ' '.join(sys.argv), 0, indent, 1474 col1_width) 1475 # We want the output to show a customized name for the pid and pgid but we want it to look like a valid 1476 # variable name. Therefore, we'll use pgm_name_var_name which was set when this module was imported. 1477 buffer += sprint_varx(pgm_name_var_name + "_pid", os.getpid(), 0, indent, 1478 col1_width) 1479 buffer += sprint_varx(pgm_name_var_name + "_pgid", os.getpgrp(), 0, indent, 1480 col1_width) 1481 userid_num = str(os.geteuid()) 1482 try: 1483 username = os.getlogin() 1484 except OSError: 1485 if userid_num == "0": 1486 username = "root" 1487 else: 1488 username = "?" 1489 buffer += sprint_varx("uid", userid_num + " (" + username 1490 + ")", 0, indent, col1_width) 1491 buffer += sprint_varx("gid", str(os.getgid()) + " (" 1492 + str(grp.getgrgid(os.getgid()).gr_name) + ")", 0, 1493 indent, col1_width) 1494 buffer += sprint_varx("host_name", socket.gethostname(), 0, indent, 1495 col1_width) 1496 try: 1497 DISPLAY = os.environ['DISPLAY'] 1498 except KeyError: 1499 DISPLAY = "" 1500 buffer += sprint_var(DISPLAY, 0, indent, col1_width) 1501 PYTHON_VERSION = os.environ.get('PYTHON_VERSION', None) 1502 if PYTHON_VERSION is not None: 1503 buffer += sprint_var(PYTHON_VERSION, 0, indent, col1_width) 1504 PYTHON_PGM_PATH = os.environ.get('PYTHON_PGM_PATH', None) 1505 if PYTHON_PGM_PATH is not None: 1506 buffer += sprint_var(PYTHON_PGM_PATH, 0, indent, col1_width) 1507 python_version = sys.version.replace("\n", "") 1508 buffer += sprint_var(python_version, 0, indent, col1_width) 1509 ROBOT_VERSION = os.environ.get('ROBOT_VERSION', None) 1510 if ROBOT_VERSION is not None: 1511 buffer += sprint_var(ROBOT_VERSION, 0, indent, col1_width) 1512 ROBOT_PGM_PATH = os.environ.get('ROBOT_PGM_PATH', None) 1513 if ROBOT_PGM_PATH is not None: 1514 buffer += sprint_var(ROBOT_PGM_PATH, 0, indent, col1_width) 1515 1516 # __builtin__.arg_obj is created by the get_arg module function, gen_get_options. 1517 try: 1518 buffer += ga.sprint_args(__builtin__.arg_obj, indent) 1519 except AttributeError: 1520 pass 1521 1522 if robot_env: 1523 # Get value of global parm_list. 1524 parm_list = BuiltIn().get_variable_value("${parm_list}") 1525 1526 for parm in parm_list: 1527 parm_value = BuiltIn().get_variable_value("${" + parm + "}") 1528 buffer += sprint_varx(parm, parm_value, 0, indent, col1_width) 1529 1530 # Setting global program_pid. 1531 BuiltIn().set_global_variable("${program_pid}", os.getpid()) 1532 1533 if linefeed: 1534 buffer += "\n" 1535 1536 return buffer 1537 1538 1539def sprint_error_report(error_text="\n", 1540 indent=2, 1541 format=None, 1542 stack_frame_ix=None): 1543 r""" 1544 Return a string with a standardized report which includes the caller's error text, the call stack and the 1545 program header. 1546 1547 Description of argument(s): 1548 error_text The error text to be included in the report. The caller should include 1549 any needed linefeeds. 1550 indent The number of characters to indent each line of output. 1551 format Long or short format. Long includes extras like lines of dashes, call 1552 stack, etc. 1553 stack_frame_ix The index of the first stack frame which is to be shown in the 1554 print_call_stack portion of the error report. 1555 """ 1556 1557 # Process input. 1558 indent = int(indent) 1559 if format is None: 1560 if robot_env: 1561 format = 'short' 1562 else: 1563 format = 'long' 1564 error_text = error_text.rstrip('\n') + '\n' 1565 1566 if format == 'short': 1567 return sprint_error(error_text) 1568 1569 buffer = "" 1570 buffer += sprint_dashes(width=120, char="=") 1571 buffer += sprint_error(error_text) 1572 buffer += "\n" 1573 if not stack_frame_ix: 1574 # Calling sprint_call_stack with stack_frame_ix of 0 causes it to show itself and this function in 1575 # the call stack. This is not helpful to a debugger and is therefore clutter. We will adjust the 1576 # stack_frame_ix to hide that information. 1577 stack_frame_ix = 1 1578 caller_func_name = sprint_func_name(1) 1579 if caller_func_name.endswith("print_error_report"): 1580 stack_frame_ix += 1 1581 caller_func_name = sprint_func_name(2) 1582 if caller_func_name.endswith("print_error_report"): 1583 stack_frame_ix += 1 1584 buffer += sprint_call_stack(indent, stack_frame_ix) 1585 buffer += sprint_pgm_header(indent) 1586 buffer += sprint_dashes(width=120, char="=") 1587 1588 return buffer 1589 1590 1591def sprint_issuing(cmd_buf, 1592 test_mode=0): 1593 r""" 1594 Return a line indicating a command that the program is about to execute. 1595 1596 Sample output for a cmd_buf of "ls" 1597 1598 #(CDT) 2016/08/25 17:57:36 - Issuing: ls 1599 1600 Description of argument(s): 1601 cmd_buf The command to be executed by caller. 1602 test_mode With test_mode set, the output will look like this: 1603 1604 #(CDT) 2016/08/25 17:57:36 - (test_mode) Issuing: ls 1605 1606 """ 1607 1608 buffer = sprint_time() 1609 if test_mode: 1610 buffer += "(test_mode) " 1611 if type(cmd_buf) is list: 1612 # Assume this is a robot command in the form of a list. 1613 cmd_buf = ' '.join([str(element) for element in cmd_buf]) 1614 buffer += "Issuing: " + cmd_buf + "\n" 1615 1616 return buffer 1617 1618 1619def sprint_pgm_footer(): 1620 r""" 1621 Return a standardized footer that programs should print at the end of the program run. It includes 1622 useful information like total run time, etc. 1623 """ 1624 1625 buffer = "\n" + sprint_time() + "Finished running " + pgm_name + ".\n\n" 1626 1627 total_time = time.time() - start_time 1628 total_time_string = "%0.6f" % total_time 1629 1630 buffer += sprint_varx(pgm_name_var_name + "_runtime", total_time_string) 1631 buffer += "\n" 1632 1633 return buffer 1634 1635 1636def sprint_file(file_path): 1637 r""" 1638 Return the file data as a string. 1639 1640 Description of argument(s): 1641 file_path The path to a file (e.g. "/tmp/file1"). 1642 """ 1643 1644 with open(file_path, 'r') as file: 1645 buffer = file.read() 1646 return buffer 1647 1648 1649def sprint(buffer=""): 1650 r""" 1651 Simply return the user's buffer. This function is used by the qprint and dprint functions defined 1652 dynamically below, i.e. it would not normally be called for general use. 1653 1654 Description of argument(s). 1655 buffer This will be returned to the caller. 1656 """ 1657 1658 try: 1659 return str(buffer) 1660 except UnicodeEncodeError: 1661 return buffer 1662 1663 1664def sprintn(buffer=""): 1665 r""" 1666 Simply return the user's buffer with a line feed. This function is used by the qprint and dprint 1667 functions defined dynamically below, i.e. it would not normally be called for general use. 1668 1669 Description of argument(s). 1670 buffer This will be returned to the caller. 1671 """ 1672 1673 try: 1674 buffer = str(buffer) + "\n" 1675 except UnicodeEncodeError: 1676 buffer = buffer + "\n" 1677 1678 return buffer 1679 1680 1681def gp_print(buffer, 1682 stream='stdout'): 1683 r""" 1684 Print the buffer using either sys.stdout.write or BuiltIn().log_to_console depending on whether we are 1685 running in a robot environment. 1686 1687 This function is intended for use only by other functions in this module. 1688 1689 Description of argument(s): 1690 buffer The string to be printed. 1691 stream Either "stdout" or "stderr". 1692 """ 1693 1694 if robot_env: 1695 BuiltIn().log_to_console(buffer, stream=stream, no_newline=True) 1696 else: 1697 if stream == "stdout": 1698 sys.stdout.write(buffer) 1699 sys.stdout.flush() 1700 else: 1701 sys.stderr.write(buffer) 1702 sys.stderr.flush() 1703 1704 1705def gp_log(buffer): 1706 r""" 1707 Log the buffer using either python logging or BuiltIn().log depending on whether we are running in a 1708 robot environment. 1709 1710 This function is intended for use only by other functions in this module. 1711 1712 Description of argument(s): 1713 buffer The string to be logged. 1714 """ 1715 1716 if robot_env: 1717 BuiltIn().log(buffer) 1718 else: 1719 logging.warning(buffer) 1720 1721 1722def gp_debug_print(buffer): 1723 r""" 1724 Print with gp_print only if gen_print_debug is set. 1725 1726 This function is intended for use only by other functions in this module. 1727 1728 Description of argument(s): 1729 buffer The string to be printed. 1730 """ 1731 1732 if not gen_print_debug: 1733 return 1734 1735 gp_print(buffer) 1736 1737 1738def get_var_value(var_value=None, 1739 default=1, 1740 var_name=None): 1741 r""" 1742 Return either var_value, the corresponding global value or default. 1743 1744 If var_value is not None, it will simply be returned. 1745 1746 If var_value is None, this function will return the corresponding global value of the variable in 1747 question. 1748 1749 Note: For global values, if we are in a robot environment, get_variable_value will be used. Otherwise, 1750 the __builtin__ version of the variable is returned (which are set by gen_arg.py functions). 1751 1752 If there is no global value associated with the variable, default is returned. 1753 1754 This function is useful for other functions in setting default values for parameters. 1755 1756 Example use: 1757 1758 def my_func(quiet=None): 1759 1760 quiet = int(get_var_value(quiet, 0)) 1761 1762 Example calls to my_func(): 1763 1764 In the following example, the caller is explicitly asking to have quiet be set to 1. 1765 1766 my_func(quiet=1) 1767 1768 In the following example, quiet will be set to the global value of quiet, if defined, or to 0 (the 1769 default). 1770 1771 my_func() 1772 1773 Description of argument(s): 1774 var_value The value to be returned (if not equal to None). 1775 default The value that is returned if var_value is None and there is no 1776 corresponding global value defined. 1777 var_name The name of the variable whose value is to be returned. Under most 1778 circumstances, this value need not be provided. This function can figure 1779 out the name of the variable passed as var_value. One exception to this 1780 would be if this function is called directly from a .robot file. 1781 """ 1782 1783 if var_value is not None: 1784 return var_value 1785 1786 if var_name is None: 1787 var_name = get_arg_name(None, 1, 2) 1788 1789 if robot_env: 1790 var_value = BuiltIn().get_variable_value("${" + var_name + "}", 1791 default) 1792 else: 1793 var_value = getattr(__builtin__, var_name, default) 1794 1795 return var_value 1796 1797 1798def get_stack_var(var_name, 1799 default="", 1800 init_stack_ix=2): 1801 r""" 1802 Starting with the caller's stack level, search upward in the call stack for a variable named var_name and 1803 return its value. If the variable cannot be found in the stack, attempt to get the global value. If the 1804 variable still cannot be found, return default. 1805 1806 Example code: 1807 1808 def func12(): 1809 my_loc_var1 = get_stack_var('my_var1', "default value") 1810 1811 def func11(): 1812 my_var1 = 11 1813 func12() 1814 1815 In this example, get_stack_var will find the value of my_var1 in func11's stack and will therefore return 1816 the value 11. Therefore, my_loc_var1 would get set to 11. 1817 1818 Description of argument(s): 1819 var_name The name of the variable to be searched for. 1820 default The value to return if the the variable cannot be found. 1821 init_stack_ix The initial stack index from which to begin the search. 0 would be the 1822 index of this func1tion ("get_stack_var"), 1 would be the index of the 1823 function calling this function, etc. 1824 """ 1825 1826 work_around_inspect_stack_cwd_failure() 1827 default = get_var_value(var_name=var_name, default=default) 1828 return next((frame[0].f_locals[var_name] 1829 for frame in inspect.stack()[init_stack_ix:] 1830 if var_name in frame[0].f_locals), default) 1831 1832 1833# hidden_text is a list of passwords which are to be replaced with asterisks by print functions defined in 1834# this module. 1835hidden_text = [] 1836# password_regex is created based on the contents of hidden_text. 1837password_regex = "" 1838 1839 1840def register_passwords(*args): 1841 r""" 1842 Register one or more passwords which are to be hidden in output produced by the print functions in this 1843 module. 1844 1845 Note: Blank password values are NOT registered. They are simply ignored. 1846 1847 Description of argument(s): 1848 args One or more password values. If a given password value is already 1849 registered, this function will simply do nothing. 1850 """ 1851 1852 global hidden_text 1853 global password_regex 1854 1855 for password in args: 1856 if password == "": 1857 break 1858 if password in hidden_text: 1859 break 1860 1861 # Place the password into the hidden_text list. 1862 hidden_text.append(password) 1863 # Create a corresponding password regular expression. Escape regex special characters too. 1864 password_regex = '(' +\ 1865 '|'.join([re.escape(x) for x in hidden_text]) + ')' 1866 1867 1868def replace_passwords(buffer): 1869 r""" 1870 Return the buffer but with all registered passwords replaced by a string of asterisks. 1871 1872 1873 Description of argument(s): 1874 buffer The string to be returned but with passwords replaced. 1875 """ 1876 1877 global password_regex 1878 1879 if int(os.environ.get("DEBUG_SHOW_PASSWORDS", "0")): 1880 return buffer 1881 1882 if password_regex == "": 1883 # No passwords to replace. 1884 return buffer 1885 1886 return re.sub(password_regex, "********", buffer) 1887 1888 1889def create_print_wrapper_funcs(func_names, 1890 stderr_func_names, 1891 replace_dict, 1892 func_prefix=""): 1893 r""" 1894 Generate code for print wrapper functions and return the generated code as a string. 1895 1896 To illustrate, suppose there is a "print_foo_bar" function in the func_names list. 1897 This function will... 1898 - Expect that there is an sprint_foo_bar function already in existence. 1899 - Create a print_foo_bar function which calls sprint_foo_bar and prints the result. 1900 - Create a qprint_foo_bar function which calls upon sprint_foo_bar only if global value quiet is 0. 1901 - Create a dprint_foo_bar function which calls upon sprint_foo_bar only if global value debug is 1. 1902 1903 Also, code will be generated to define aliases for each function as well. Each alias will be created by 1904 replacing "print_" in the function name with "p" For example, the alias for print_foo_bar will be 1905 pfoo_bar. 1906 1907 Description of argument(s): 1908 func_names A list of functions for which print wrapper function code is to be 1909 generated. 1910 stderr_func_names A list of functions whose generated code should print to stderr rather 1911 than to stdout. 1912 replace_dict Please see the create_func_def_string function in wrap_utils.py for 1913 details on this parameter. This parameter will be passed directly to 1914 create_func_def_string. 1915 func_prefix Prefix to be pre-pended to the generated function name. 1916 """ 1917 1918 buffer = "" 1919 1920 for func_name in func_names: 1921 if func_name in stderr_func_names: 1922 replace_dict['output_stream'] = "stderr" 1923 else: 1924 replace_dict['output_stream'] = "stdout" 1925 1926 s_func_name = "s" + func_name 1927 q_func_name = "q" + func_name 1928 d_func_name = "d" + func_name 1929 1930 # We don't want to try to redefine the "print" function, thus the following if statement. 1931 if func_name != "print": 1932 func_def = create_func_def_string(s_func_name, 1933 func_prefix + func_name, 1934 print_func_template, 1935 replace_dict) 1936 buffer += func_def 1937 1938 func_def = create_func_def_string(s_func_name, 1939 func_prefix + "q" + func_name, 1940 qprint_func_template, replace_dict) 1941 buffer += func_def 1942 1943 func_def = create_func_def_string(s_func_name, 1944 func_prefix + "d" + func_name, 1945 dprint_func_template, replace_dict) 1946 buffer += func_def 1947 1948 func_def = create_func_def_string(s_func_name, 1949 func_prefix + "l" + func_name, 1950 lprint_func_template, replace_dict) 1951 buffer += func_def 1952 1953 # Create abbreviated aliases (e.g. spvar is an alias for sprint_var). 1954 alias = re.sub("print_", "p", func_name) 1955 alias = re.sub("print", "p", alias) 1956 prefixes = [func_prefix + "", "s", func_prefix + "q", 1957 func_prefix + "d", func_prefix + "l"] 1958 for prefix in prefixes: 1959 if alias == "p": 1960 continue 1961 func_def = prefix + alias + " = " + prefix + func_name 1962 buffer += func_def + "\n" 1963 1964 return buffer 1965 1966 1967# In the following section of code, we will dynamically create print versions for each of the sprint 1968# functions defined above. So, for example, where we have an sprint_time() function defined above that 1969# returns the time to the caller in a string, we will create a corresponding print_time() function that will 1970# print that string directly to stdout. 1971 1972# It can be complicated to follow what's being created below. Here is an example of the print_time() 1973# function that will be created: 1974 1975# def print_time(buffer=''): 1976# gp_print(replace_passwords(sprint_time(buffer=buffer)), stream='stdout') 1977 1978# For each print function defined below, there will also be a qprint, a dprint and an lprint version defined 1979# (e.g. qprint_time, dprint_time, lprint_time). 1980 1981# The q version of each print function will only print if the quiet variable is 0. 1982# The d version of each print function will only print if the debug variable is 1. 1983# The l version of each print function will print the contents as log data. For conventional programs, this 1984# means use of the logging module. For robot programs it means use of the BuiltIn().log() function. 1985 1986# Templates for the various print wrapper functions. 1987print_func_template = \ 1988 [ 1989 " <mod_qualifier>gp_print(<mod_qualifier>replace_passwords(" 1990 + "<call_line>), stream='<output_stream>')" 1991 ] 1992 1993qprint_func_template = \ 1994 [ 1995 " quiet = <mod_qualifier>get_stack_var(\"quiet\", 0)", 1996 " if int(quiet): return" 1997 ] + print_func_template 1998 1999dprint_func_template = \ 2000 [ 2001 " debug = <mod_qualifier>get_stack_var(\"debug\", 0)", 2002 " if not int(debug): return" 2003 ] + print_func_template 2004 2005lprint_func_template = \ 2006 [ 2007 " <mod_qualifier>set_last_seconds_ix(<mod_qualifier>" 2008 + "lprint_last_seconds_ix())", 2009 " <mod_qualifier>gp_log(<mod_qualifier>replace_passwords" 2010 + "(<call_line>))", 2011 " <mod_qualifier>set_last_seconds_ix(<mod_qualifier>" 2012 + "standard_print_last_seconds_ix())" 2013 ] 2014 2015replace_dict = {'output_stream': 'stdout', 'mod_qualifier': ''} 2016 2017gp_debug_print("robot_env: " + str(robot_env) + "\n") 2018 2019# func_names contains a list of all print functions which should be created from their sprint counterparts. 2020func_names = ['print_time', 'print_timen', 'print_error', 'print_varx', 2021 'print_var', 'print_vars', 'print_dashes', 'indent', 2022 'print_call_stack', 'print_func_name', 'print_executing', 2023 'print_pgm_header', 'print_issuing', 'print_pgm_footer', 2024 'print_file', 'print_error_report', 'print', 'printn'] 2025 2026# stderr_func_names is a list of functions whose output should go to stderr rather than stdout. 2027stderr_func_names = ['print_error', 'print_error_report'] 2028 2029func_defs = create_print_wrapper_funcs(func_names, stderr_func_names, 2030 replace_dict) 2031gp_debug_print(func_defs) 2032exec(func_defs) 2033