#!/usr/bin/wish # This file provides many valuable print procedures such as sprint_var, sprint_time, sprint_error, etc. my_source [list data_proc.tcl call_stack.tcl] # Need "Expect" package for trap procedure. package require Expect # Setting the following variables for use both inside this file and by programs sourcing this file. set program_path $argv0 set program_dir_path "[file dirname $argv0]/" set program_name "[file tail $argv0]" # Some procedures (e.g. sprint_pgm_header) need a program name value that looks more like a valid variable # name. Therefore, we'll swap out odd characters (like ".") for underscores. regsub {\.} $program_name "_" pgm_name_var_name # Initialize some time variables used in procedures in this file. set start_time [clock microseconds] proc calc_wrap_stack_ix_adjust {} { # Calculate and return a number which can be used as an offset into the call stack for wrapper procedures. # NOTE: This procedure is designed expressly to work with this file's print procedures scheme (i.e. # print_x is a wrapper for sprint_x, etc.). In other words, this procedure may not be well-suited for # general use. # Get a list of the procedures in the call stack beginning with our immediate caller on up to the # top-level caller. set call_stack [get_call_stack -2] # The first stack entry is our immediate caller. set caller [lindex $call_stack 0] # Remove first entry from stack. set call_stack [lreplace $call_stack 0 0] # Strip any leading "s" to arrive at base_caller name (e.g. the corresponding base name for "sprint_var" # would be "print_var"). set base_caller [string trimleft $caller s] # Account for alias print procedures which have "p" vs "print_" (e.g. pvar vs print_var). regsub "print_" $base_caller "p" alias_base_caller # Initialize the stack_ix_adjust value. set stack_ix_adjust 0 # Note: print_vars|pvars is a special case so we add it explicitly to the regex below. set regex ".*(${base_caller}|${alias_base_caller}|print_vars|pvars)$" foreach proc_name $call_stack { # For every remaining stack item that looks like a wrapper (i.e. matches our regex), we increment the # stack_ix_adjust. if { [regexp -expanded $regex $proc_name]} { incr stack_ix_adjust continue } # If there is no match, then we are done. break } return $stack_ix_adjust } # hidden_text is a list of passwords which are to be replaced with asterisks by print procedures defined in # this file. set hidden_text [list] # password_regex is created from the contents of the hidden_text list above. set password_regex "" proc register_passwords {args} { # Register one or more passwords which are to be hidden in output produced by the print procedures in this # file. # Note: Blank password values are NOT registered. They are simply ignored. # Description of argument(s): # args One or more password values. If a given password value is already # registered, this procedure will simply ignore it, i.e. there will be no # duplicate values in the hidden_text list. global hidden_text global password_regex foreach password $args { # Skip blank passwords. if { $password == "" } { continue } # Skip already-registered passwords. if { [lsearch -exact $hidden_text $password] != -1 } { continue } # Put the password into the global hidden_text list. lappend hidden_text $password } # TODO: Escape metachars in the password_regex. set password_regex [join $hidden_text |] } proc replace_passwords {buffer} { # Replace all registered password found in buffer with a string of asterisks and return the result. # Description of argument(s): # buffer The string to be altered and returned. # Note: If environment variable GEN_PRINT_DEBUG is set, this procedure will do nothing. global env if { [get_var ::env(GEN_PRINT_DEBUG) 0] } { return $buffer } if { [get_var ::env(DEBUG_SHOW_PASSWORDS) 0] } { return $buffer } global password_regex # No passwords to replace? if { $password_regex == "" } { return $buffer } regsub -all "${password_regex}" $buffer {********} buffer return $buffer } proc my_time { cmd_buf { iterations 100 } } { # Run the "time" function on the given command string and print the results. # The main benefit of running this vs just doing the "time" command directly: # - This will print the results. # Description of argument(s): # cmd_buf The command string to be run. # iterations The number of times to run the command string. Typically, more # iterations yields more accurate results. print_issuing $cmd_buf set result [time {uplevel 1 $cmd_buf} $iterations] set raw_microseconds [lindex [split [lindex $result 0] .] 0] set seconds [expr $raw_microseconds / 1000000] set raw_microseconds [expr $raw_microseconds % 1000000] set seconds_per_iteration [format "%i.%06i" ${seconds}\ ${raw_microseconds}] print_var seconds_per_iteration } # If environment variable "GEN_PRINT_DEBUG" is set, this module will output debug data. This is primarily # intended for the developer of this module. set GEN_PRINT_DEBUG [get_var ::env(GEN_PRINT_DEBUG) 0] # The user can set the following environment variables to influence the output from print_time and print_var # procedures. See the prologs of those procedures for details. set NANOSECONDS [get_var ::env(NANOSECONDS) 0] set SHOW_ELAPSED_TIME [get_var ::env(SHOW_ELAPSED_TIME) 0] # _gtp_default_print_var_width_ is adjusted based on NANOSECONDS and SHOW_ELAPSED_TIME. if { $NANOSECONDS } { set _gtp_default_print_var_width_ 36 set width_incr 14 } else { set _gtp_default_print_var_width_ 29 set width_incr 7 } if { $SHOW_ELAPSED_TIME } { incr _gtp_default_print_var_width_ $width_incr # Initializing _sprint_time_last_seconds_ which is a global value to remember the clock seconds from the # last time sprint_time was called. set _gtp_sprint_time_last_micro_seconds_ [clock microseconds] } # tcl_precision is a built-in Tcl variable that specifies the number of digits to generate when converting # floating-point values to strings. set tcl_precision 17 proc sprint { { buffer {} } } { # Simply return the user's buffer. # This procedure is used by the qprint and dprint functions defined dynamically below, i.e. it would not # normally be called for general use. # Description of arguments. # buffer This will be returned to the caller. return $buffer } proc sprintn { { buffer {} } } { # Simply return the user's buffer plus a trailing line feed.. # This procedure is used by the qprintn and dprintn functions defined dynamically below, i.e. it would not # normally be called for general use. # Description of arguments. # buffer This will be returned to the caller. return ${buffer}\n } proc sprint_time { { buffer {} } } { # Return the time in a formatted manner as described below. # Example: # The following tcl code... # puts -nonewline [sprint_time()] # puts -nonewline ["Hi.\n"] # Will result in the following type of output: # #(CDT) 2016/07/08 15:25:35 - Hi. # Example: # The following tcl code... # puts -nonewline [sprint_time("Hi.\n")] # Will result in the following type of output: # #(CDT) 2016/08/03 17:12:05 - Hi. # The following environment variables will affect the formatting as described: # NANOSECONDS This will cause the time stamps to be precise to the microsecond (Yes, it # probably should have been named MICROSECONDS but the convention was set # long ago so we're sticking with it). Example of the output when # environment variable NANOSECONDS=1. # #(CDT) 2016/08/03 17:16:25.510469 - Hi. # SHOW_ELAPSED_TIME This will cause the elapsed time to be included in the output. This is # the amount of time that has elapsed since the last time this procedure # was called. The precision of the elapsed time field is also affected by # the value of the NANOSECONDS environment variable. Example of the output # when environment variable NANOSECONDS=0 and SHOW_ELAPSED_TIME=1. # #(CDT) 2016/08/03 17:17:40 - 0 - Hi. # Example of the output when environment variable NANOSECONDS=1 and SHOW_ELAPSED_TIME=1. # #(CDT) 2016/08/03 17:18:47.317339 - 0.000046 - Hi. # Description of argument(s). # buffer A string string whhich is to be appended to the formatted time string and # returned. global NANOSECONDS global _gtp_sprint_time_last_micro_seconds_ global SHOW_ELAPSED_TIME # Get micro seconds since the epoch. set epoch_micro [clock microseconds] # Break the left and right of the decimal point. set epoch_seconds [expr $epoch_micro / 1000000] set epoch_decimal_micro [expr $epoch_micro % 1000000] set format_string "#(%Z) %Y/%m/%d %H:%M:%S" set return_string [clock format $epoch_seconds -format\ "#(%Z) %Y/%m/%d %H:%M:%S"] if { $NANOSECONDS } { append return_string ".[format "%06i" ${epoch_decimal_micro}]" } if { $SHOW_ELAPSED_TIME } { set return_string "${return_string} - " set elapsed_micro [expr $epoch_micro - \ $_gtp_sprint_time_last_micro_seconds_] set elapsed_seconds [expr $elapsed_micro / 1000000] if { $NANOSECONDS } { set elapsed_decimal_micro [expr $elapsed_micro % 1000000] set elapsed_float [format "%i.%06i" ${elapsed_seconds}\ ${elapsed_decimal_micro}] set elapsed_time_buffer "[format "%11.6f" ${elapsed_float}]" } else { set elapsed_time_buffer "[format "%4i" $elapsed_seconds]" } set return_string "${return_string}${elapsed_time_buffer}" } set return_string "${return_string} - ${buffer}" set _gtp_sprint_time_last_micro_seconds_ $epoch_micro return $return_string } proc sprint_timen { args } { # Return the value of sprint_time + a line feed. # Description of argument(s): # args All args are passed directly to subordinate function, sprint_time. See # that function's prolog for details. return [sprint_time {*}$args]\n } proc sprint_error { { buffer {} } } { # Return a standardized error string which includes the callers buffer text. # Description of argument(s): # buffer Text to be returned as part of the error message. return [sprint_time "**ERROR** $buffer"] } proc sprint_varx { var_name var_value { indent 0 } { width {} } { hex 0 } } { # Return the name and value of the variable named in var_name in a formatted way. # This procedure will visually align the output to look good next to print_time output. # Example: # Given the following code: # print_timen "Initializing variables." # set first_name "Joe" # set last_name "Montana" # set age 50 # print_varx last_name $last_name # print_varx first_name $first_name 2 # print_varx age $age 2 # With environment variables NANOSECONDS and SHOW_ELAPSED_TIME both set, the following output is produced: # #(CST) 2017/12/14 16:38:28.259480 - 0.000651 - Initializing variables. # last_name: Montana # first_name: Joe # age: 50 # Description of argument(s): # var_name The name of the variable whose name and value are to be printed. # var_value The value to be printed. # indent The number of spaces to indent each line of output. # width The width of the column containing the variable name. By default this # will align with the print_time text (see example above). # hex Indicates that the variable value is to be printed in hexedecimal format. # This is only valid if the variable value is an integer. If the variable # is NOT an integer and is blank, this will be interpreted to mean "print # the string '', rather than an actual blank value". # Note: This procedure relies on global var _gtp_default_print_var_width_ set_var_default indent 0 global _gtp_default_print_var_width_ set_var_default width $_gtp_default_print_var_width_ if { $indent > 0 } { set width [expr $width - $indent] } if { $hex } { if { [catch {format "0x%08x" "$var_value"} result] } { if { $var_value == "" } { set var_value "" } set hex 0 } } if { $hex } { append buffer "[format "%-${indent}s%-${width}s0x%08x" "" "$var_name:" \ "$var_value"]" } else { append buffer "[format "%-${indent}s%-${width}s%s" "" "$var_name:" \ "$var_value"]" } return $buffer\n } proc sprint_var { var_name args } { # Return the name and value of the variable named in var_name in a formatted way. # This procedure will visually align the output to look good next to print_time output. # Note: This procedure is the equivalent of sprint_varx with one difference: This function will figure # out the value of the named variable whereas sprint_varx expects you to pass the value. This procedure in # fact calls sprint_varx to do its work. # Note: This procedure will detect whether var_name is an array and print it accordingly (see the second # example below). # Example: # Given the following code: # print_timen "Initializing variables." # set first_name "Joe" # set last_name "Montana" # set age 50 # print_var last_name # print_var first_name 2 # print_var age 2 # With environment variables NANOSECONDS and SHOW_ELAPSED_TIME both set, the following output is produced: # #(CST) 2017/12/14 16:38:28.259480 - 0.000651 - Initializing variables. # last_name: Montana # first_name: Joe # age: 50 # Example: # Given the following code: # set data(0) cow # set data(1) horse # print_var data # data: # data(0): cow # data(1): horse # Description of argument(s): # var_name The name of the variable whose name and value are to be printed. # args The args understood by sprint_varx (after var_name and var_value). See # sprint_varx's prolog for details. # Note: This procedure relies on global var _gtp_default_print_var_width_ # Determine who our caller is and therefore what upvar_level to use to get var_value. set stack_ix_adjust [calc_wrap_stack_ix_adjust] set upvar_level [expr $stack_ix_adjust + 1] upvar $upvar_level $var_name var_value # Special processing for arrays: if { [array exists var_value] } { set indent [lindex $args 0] set args [lrange $args 1 end] set_var_default indent 0 append buffer [format "%-${indent}s%s\n" "" "$var_name:"] incr indent 2 incr width -2 set search_token [array startsearch var_value] while {[array anymore var_value $search_token]} { set key [array nextelement var_value $search_token] set arr_value $var_value($key) append buffer [sprint_varx "${var_name}(${key})" $arr_value $indent\ {*}$args] } array donesearch var_value $search_token return $buffer } # If var_value is not defined, catch the error and print its value as "variable not set". if {[catch {set buffer [sprint_varx $var_name $var_value {*}$args]} error_text options]} { set regex ":\[ \]no\[ \]such\[ \]variable" if { [regexp -expanded ${regex} ${error_text}]} { return [sprint_varx $var_name {** variable not set **} {*}$args] } else { print_dict options exit 1 } } else { return $buffer } } proc sprint_list { var_name args } { # Return the name and value of the list variable named in var_name in a formatted way. # This procedure is the equivalent of sprint_var but for lists. # Description of argument(s): # var_name The name of the variable whose name and value are to be printed. # args The args understood by sprint_varx (after var_name and var_value). See # sprint_varx's prolog for details. # Note: In TCL, there is no way to determine that a variable represents a list vs a string, etc. It is up # to the programmer to decide how the data is to be interpreted. Thus the need for procedures such as this # one. Consider the following code: # set my_list {one two three} # print_var my_list # print_list my_list # Output from aforementioned code: # my_list: one two three # my_list: # my_list[0]: one # my_list[1]: two # my_list[2]: three # As far as print_var is concerned, my_list is a string and is printed accordingly. By using print_list, # the programmer is asking to have the output shown as a list with list indices, etc. # Determine who our caller is and therefore what upvar_level to use. set stack_ix_adjust [calc_wrap_stack_ix_adjust] set upvar_level [expr $stack_ix_adjust + 1] upvar $upvar_level $var_name var_value set indent [lindex $args 0] set args [lrange $args 1 end] set_var_default indent 0 append buffer [format "%-${indent}s%s\n" "" "$var_name:"] incr indent 2 set index 0 foreach element $var_value { append buffer [sprint_varx "${var_name}\[${index}\]" $element $indent\ {*}$args] incr index } return $buffer } proc sprint_dict { var_name args } { # Return the name and value of the dictionary variable named in var_name in a formatted way. # This procedure is the equivalent of sprint_var but for dictionaries. # Description of argument(s): # var_name The name of the variable whose name and value are to be printed. # args The args understood by sprint_varx (after var_name and var_value). See # sprint_varx's prolog for details. # Note: In TCL, there is no way to determine that a variable represents a dictionary vs a string, etc. It # is up to the programmer to decide how the data is to be interpreted. Thus the need for procedures such # as this one. Consider the following code: # set my_dict [dict create first Joe last Montana age 50] # print_var my_dict # print_dict my_dict # Output from aforementioned code: # my_dict: first Joe last Montana age 50 # my_dict: # my_dict[first]: Joe # my_dict[last]: Montana # my_dict[age]: 50 # As far as print_var is concerned, my_dict is a string and is printed accordingly. By using print_dict, # the programmer is asking to have the output shown as a dictionary with dictionary keys/values, etc. # Determine who our caller is and therefore what upvar_level to use. set stack_ix_adjust [calc_wrap_stack_ix_adjust] set upvar_level [expr $stack_ix_adjust + 1] upvar $upvar_level $var_name var_value set indent [lindex $args 0] set args [lrange $args 1 end] set_var_default indent 0 append buffer [format "%-${indent}s%s\n" "" "$var_name:"] incr indent 2 foreach {key value} $var_value { append buffer [sprint_varx "${var_name}\[${key}\]" $value $indent {*}$args] incr index } return $buffer } proc sprint_vars { args } { # Sprint the values of one or more variables. # Description of arg(s): # args: A list of variable names to be printed. The first argument in the arg list found to be an # integer (rather than a variable name) will be interpreted to be first of several possible sprint_var # arguments (e.g. indent, width, hex). See the prologue for sprint_var above for descriptions of this # variables. # Example usage: # set var1 "hello" # set var2 "there" # set indent 2 # set buffer [sprint_vars var1 var2] # or... # set buffer [sprint_vars var1 var2 $indent] # Look for integer arguments. set first_int_ix [lsearch -regexp $args {^[0-9]+$}] if { $first_int_ix == -1 } { # If none are found, sub_args is set to empty. set sub_args {} } else { # Set sub_args to the portion of the arg list that are integers. set sub_args [lrange $args $first_int_ix end] # Re-set args to exclude the integer values. set args [lrange $args 0 [expr $first_int_ix - 1]] } foreach arg $args { append buffer [sprint_var $arg {*}$sub_args] } return $buffer } proc sprint_dashes { { indent 0 } { width 80 } { line_feed 1 } { char "-" } } { # Return a string of dashes to the caller. # Description of argument(s): # indent The number of characters to indent the output. # width The width of the string of dashes. # line_feed Indicates whether the output should end with a line feed. # char The character to be repeated in the output string. In other words, you # can call on this function to print a string of any character (e.g. "=", # "_", etc.). set_var_default indent 0 set_var_default width 80 set_var_default line_feed 1 append buffer [string repeat " " $indent][string repeat $char $width] append buffer [string repeat "\n" $line_feed] return $buffer } proc sprint_executing {{ include_args 1 }} { # Return a string that looks something like this: # #(CST) 2017/11/28 15:08:03.261466 - 0.015214 - Executing: proc1 hi # Description of argument(s): # include_args Indicates whether proc args should be included in the result. set stack_ix_adjust [calc_wrap_stack_ix_adjust] set level [expr -(2 + $stack_ix_adjust)] return "[sprint_time]Executing: [get_stack_proc_name $level $include_args]\n" } proc sprint_issuing { { cmd_buf "" } { test_mode 0 } } { # Return a line indicating a command that the program is about to execute. # Sample output for a cmd_buf of "ls" # #(CDT) 2016/08/25 17:57:36 - Issuing: ls # Description of arg(s): # cmd_buf The command to be executed by caller. If this is blank, this procedure # will search up the stack for the first cmd_buf value to use. # test_mode With test_mode set, your output will look like this: # #(CDT) 2016/08/25 17:57:36 - (test_mode) Issuing: ls if { $cmd_buf == "" } { set cmd_buf [get_stack_var cmd_buf {} 2] } append buffer [sprint_time] if { $test_mode } { append buffer "(test_mode) " } append buffer "Issuing: ${cmd_buf}\n" return $buffer } proc sprint_call_stack { { indent 0 } } { # Return a call stack report for the given point in the program with line numbers, procedure names and # procedure parameters and arguments. # Sample output: # --------------------------------------------------------------------------- # TCL procedure call stack # Line # Procedure name and arguments # ------ -------------------------------------------------------------------- # 21 print_call_stack # 32 proc1 257 # --------------------------------------------------------------------------- # Description of arguments: # indent The number of characters to indent each line of output. append buffer "[sprint_dashes ${indent}]" append buffer "[string repeat " " $indent]TCL procedure call stack\n\n" append buffer "[string repeat " " $indent]" append buffer "Line # Procedure name and arguments\n" append buffer "[sprint_dashes $indent 6 0] [sprint_dashes 0 73]" for {set ix [expr [info level]-1]} {$ix > 0} {incr ix -1} { set frame_dict [info frame $ix] set line_num [dict get $frame_dict line] set proc_name_plus_args [dict get $frame_dict cmd] append buffer [format "%-${indent}s%6i %s\n" "" $line_num\ $proc_name_plus_args] } append buffer "[sprint_dashes $indent]" return $buffer } proc sprint_tcl_version {} { # Return the name and value of tcl_version in a formatted way. global tcl_version return [sprint_var tcl_version] } proc sprint_error_report { { error_text "\n" } { indent 0 } } { # Return a string with a standardized report which includes the caller's error text, the call stack and # the program header. # Description of arg(s): # error_text The error text to be included in the report. The caller should include # any needed linefeeds. # indent The number of characters to indent each line of output. set width 120 set char "=" set line_feed 1 append buffer [sprint_dashes $indent $width $line_feed $char] append buffer [string repeat " " $indent][sprint_error $error_text] append buffer "\n" append buffer [sprint_call_stack $indent] append buffer [sprint_pgm_header $indent] append buffer [sprint_dashes $indent $width $line_feed $char] return $buffer } proc sprint_pgm_header { {indent 0} {linefeed 1} } { # Return a standardized header that programs should print at the beginning of the run. It includes useful # information like command line, pid, userid, program parameters, etc. # Description of arguments: # indent The number of characters to indent each line of output. # linefeed Indicates whether a line feed be included at the beginning and end of the # report. global program_name global pgm_name_var_name global argv0 global argv global env global _gtp_default_print_var_width_ set_var_default indent 0 set indent_str [string repeat " " $indent] set width [expr $_gtp_default_print_var_width_ + $indent] # Get variable values for output. set command_line "$argv0 $argv" set pid_var_name ${pgm_name_var_name}_pid set $pid_var_name [pid] set uid [get_var ::env(USER) 0] set host_name [get_var ::env(HOSTNAME) 0] set DISPLAY [get_var ::env(DISPLAY) 0] # Generate the report. if { $linefeed } { append buffer "\n" } append buffer ${indent_str}[sprint_timen "Running ${program_name}."] append buffer ${indent_str}[sprint_timen "Program parameter values, etc.:\n"] append buffer [sprint_var command_line $indent $width] append buffer [sprint_var $pid_var_name $indent $width] append buffer [sprint_var uid $indent $width] append buffer [sprint_var host_name $indent $width] append buffer [sprint_var DISPLAY $indent $width] # Print caller's parm names/values. global longoptions global pos_parms regsub -all ":" "${longoptions} ${pos_parms}" {} parm_names foreach parm_name $parm_names { set cmd_buf "global $parm_name ; append buffer" append cmd_buf " \[sprint_var $parm_name $indent $width\]" eval $cmd_buf } if { $linefeed } { append buffer "\n" } return $buffer } proc sprint_pgm_footer {} { # Return a standardized footer that programs should print at the end of the program run. It includes # useful information like total run time, etc. global program_name global pgm_name_var_name global start_time # Calculate total runtime. set total_time_micro [expr [clock microseconds] - $start_time] # Break the left and right of the decimal point. set total_seconds [expr $total_time_micro / 1000000] set total_decimal_micro [expr $total_time_micro % 1000000] set total_time_float [format "%i.%06i" ${total_seconds}\ ${total_decimal_micro}] set total_time_string [format "%0.6f" $total_time_float] set runtime_var_name ${pgm_name_var_name}_runtime set $runtime_var_name $total_time_string append buffer [sprint_timen "Finished running ${program_name}."] append buffer "\n" append buffer [sprint_var $runtime_var_name] append buffer "\n" return $buffer } proc sprint_arg_desc { arg_title arg_desc { indent 0 } { col1_width 25 }\ { line_width 80 } } { # Return a formatted argument description. # Example: # # set desc "When in the Course of human events, it becomes necessary for one people to dissolve the # political bands which have connected them with another, and to assume among the powers of the earth, the # separate and equal station to which the Laws of Nature and of Nature's God entitle them, a decent respect # to the opinions of mankind requires that they should declare the causes which impel them to the # separation." # set buffer [sprint_arg_desc "--declaration" $desc] # puts $buffer # Resulting output: # --declaration When in the Course of human events, it becomes # necessary for one people to dissolve the # political bands which have connected them with # another, and to assume among the powers of the # earth, the separate and equal station to which # the Laws of Nature and of Nature's God entitle # them, a decent respect to the opinions of mankind # requires that they should declare the causes # which impel them to the separation. # Description of argument(s): # arg_title The content that you want to appear on the first line in column 1. # arg_desc The text that describes the argument. # indent The number of characters to indent. # col1_width The width of column 1, which is the column containing the arg_title. # line_width The total max width of each line of output. set fold_width [expr $line_width - $col1_width] set escaped_arg_desc [escape_bash_quotes "${arg_desc}"] set cmd_buf "echo '${escaped_arg_desc}' | fold --spaces --width=" append cmd_buf "${fold_width} | sed -re 's/\[ \]+$//g'" set out_buf [eval exec bash -c {$cmd_buf}] set help_lines [split $out_buf "\n"] set buffer {} set line_num 1 foreach help_line $help_lines { if { $line_num == 1 } { if { [string length $arg_title] > $col1_width } { # If the arg_title is already wider than column1, print it on its own # line. append buffer [format "%${indent}s%-${col1_width}s\n" ""\ "$arg_title"] append buffer [format "%${indent}s%-${col1_width}s%s\n" "" ""\ "${help_line}"] } else { append buffer [format "%${indent}s%-${col1_width}s%s\n" ""\ "$arg_title" "${help_line}"] } } else { append buffer [format "%${indent}s%-${col1_width}s%s\n" "" ""\ "${help_line}"] } incr line_num } return $buffer } # Define the create_print_wrapper_procs to help us create print wrappers. # First, create templates. # Notes: # - The resulting procedures will replace all registered passwords. # - The resulting "quiet" and "debug" print procedures will search the stack for quiet and debug, # respectively. That means that the if a procedure calls qprint_var and the procedure has a local version # of quiet set to 1, the print will not occur, even if there is a global version of quiet set to 0. set print_proc_template " puts -nonewline \[replace_passwords" append print_proc_template " \[ {*}\$args\]\]\n}\n" set qprint_proc_template " set quiet \[get_stack_var quiet 0\]\n if {" append qprint_proc_template " \$quiet } { return }\n${print_proc_template}" set dprint_proc_template " set debug \[get_stack_var debug 0\]\n if { !" append dprint_proc_template " \$debug } { return }\n${print_proc_template}" # Put each template into the print_proc_templates array. set print_proc_templates(p) $print_proc_template set print_proc_templates(q) $qprint_proc_template set print_proc_templates(d) $dprint_proc_template proc create_print_wrapper_procs {proc_names {stderr_proc_names {}} } { # Generate code for print wrapper procs and return the generated code as a string. # To illustrate, suppose there is a "print_foo_bar" proc in the proc_names list. # This proc will... # - Expect that there is an sprint_foo_bar proc already in existence. # - Create a print_foo_bar proc which calls sprint_foo_bar and prints the result. # - Create a qprint_foo_bar proc which calls upon sprint_foo_bar only if global value quiet is 0. # - Create a dprint_foo_bar proc which calls upon sprint_foo_bar only if global value debug is 1. # Also, code will be generated to define aliases for each proc as well. Each alias will be created by # replacing "print_" in the proc name with "p" For example, the alias for print_foo_bar will be pfoo_bar. # Description of argument(s): # proc_names A list of procs for which print wrapper proc code is to be generated. # stderr_proc_names A list of procs whose generated code should print to stderr rather than # to stdout. global print_proc_template global print_proc_templates foreach proc_name $proc_names { if { [expr [lsearch $stderr_proc_names $proc_name] == -1] } { set replace_dict(output_stream) "" } else { set replace_dict(output_stream) " stderr" } set base_proc_name "s${proc_name}" set replace_dict(base_proc_name) $base_proc_name set wrap_proc_names(p) $proc_name set wrap_proc_names(q) q${proc_name} set wrap_proc_names(d) d${proc_name} foreach template_key [list p q d] { set wrap_proc_name $wrap_proc_names($template_key) set call_line "proc ${wrap_proc_name} \{args\} \{\n" set proc_body $print_proc_templates($template_key) set proc_def ${call_line}${proc_body} foreach {key value} [array get replace_dict] { regsub -all "<$key>" $proc_def $value proc_def } regsub "print_" $wrap_proc_name "p" alias_proc_name regsub "${wrap_proc_name}" $proc_def $alias_proc_name alias_def append buffer "${proc_def}${alias_def}" } } return $buffer } # Get this file's path. set frame_dict [info frame 0] set file_path [dict get $frame_dict file] # Get a list of this file's sprint procs. set sprint_procs [get_file_proc_names $file_path sprint] # Create a corresponding list of print_procs. set proc_names [list_map $sprint_procs {[string range $x 1 end]}] # Sort them for ease of debugging. set proc_names [lsort $proc_names] set stderr_proc_names [list print_error print_error_report] set proc_def [create_print_wrapper_procs $proc_names $stderr_proc_names] if { $GEN_PRINT_DEBUG } { puts $proc_def } eval "${proc_def}"