package PBS::Debug ; use 5.006 ; use strict ; use warnings ; use Data::Dumper ; use Data::TreeDumper ; use Carp ; require Exporter ; use AutoLoader qw(AUTOLOAD) ; our @ISA = qw(Exporter) ; our %EXPORT_TAGS = ('all' => [ qw() ]) ; our @EXPORT_OK = ( @{ $EXPORT_TAGS{'all'} } ) ; our @EXPORT = qw( EnableDebugger AddBreakpoint RemoveBreakpoints ListBreakpoints ActivateBreakpoints DeactivateBreakpoints ActivatePerlDebugger DeactivatePerlDebugger CheckBreakpoint ) ; our $VERSION = '0.03' ; use PBS::PBS ; use PBS::Output ; use PBS::Information ; #------------------------------------------------------------------------------- our $debug_enabled = 0 ; # display the banner when running under the perl debugger. PrintBanner() if(*DB::DB{CODE}) ; #------------------------------------------------------------------------------- sub EnableDebugger { my $switch = shift ; my $file= shift ; # can contains breakpoint definitions if($file ne '') { unless (my $return = do $file ) { die ERROR "couldn't parse '$file ': $@" if $@; die ERROR "couldn't do '$file ': $!" unless defined $return; die ERROR "couldn't run '$file '" unless $return; } $debug_enabled = 1 ; } } #------------------------------------------------------------------------------- sub PrintBanner { PrintDebug "Pbs support available, type 'p_h' for help.\n\n" ; } push @EXPORT, 'p_h' ; sub p_h { PrintDebug < ; #~ chomp $data ; #~ return if $data eq '' ; #~ PrintDebug "breakpoint set at '$data'" ; #~ } push @EXPORT, 'p_list_breakpoints' ; *p_list_breakpoints = \&ListBreakpoints ; push @EXPORT, 'p_L' ; *p_L = \&ListBreakpoints ; #------------------------------------------------------------------------------- push @EXPORT, 'p_activate_breakpoints' ; *p_activate_breakpoints = \&ActivateBreakpoints ; push @EXPORT, 'p_ab' ; *p_ab= \&ActivateBreakpoints ; push @EXPORT, 'p_deactivate_breakpoints' ; *p_deactivate_breakpoints = \&DeactivateBreakpoints ; push @EXPORT, 'p_db' ; *p_db= \&DeactivateBreakpoints ; #------------------------------------------------------------------------------- push @EXPORT, 'p_jump_in_debugger' ; *p_jump_in_debugger= \&ActivatePerlDebugger ; push @EXPORT, 'p_j' ; *p_j = \&ActivatePerlDebugger ; push @EXPORT, 'p_no_jump_in_debugger' ; *p_no_jump_in_debugger = \&DeactivatePerlDebugger ; push @EXPORT, 'p_nj' ; *p_nj = \&DeactivatePerlDebugger ; #------------------------------------------------------------------------------- push @EXPORT, 'p_B' ; *p_B = \&RemoveBreakpoints ; #------------------------------------------------------------------------------- push @EXPORT, 'p_list' ; sub p_list { my $tree = shift ; my $node_regex = shift || '.*' ; my $package_regex = shift || '.*' ; if(!defined $tree || $tree eq 'HASH') { PrintDebug "p_list is expecting a reference to \$insereted_files\n" ; return() ; } NodeChoice($tree, $node_regex) ; } push @EXPORT, 'p_l' ; *p_l = \&p_list ; #---------------------------------------------------------------------- push @EXPORT, 'p_tree' ; sub p_tree { my $tree = shift ; my $filter = shift || \&Data::TreeDumper::HashKeysSorter ; my $indentation = shift || '' ; unless(ref $tree eq 'HASH') { PrintDebug "p_tree is expecting a reference to a tree\n" ; return() ; } PrintDebug ( DumpTree ( $tree , "Tree '$tree->{__NAME}': " , FILTER => $filter , INDENTATION => $indentation ) ) ; } push @EXPORT, 'p_t' ; *p_t = \&p_tree ; #---------------------------------------------------------------------- push @EXPORT, 'p_dependencies' ; sub p_dependencies { my $tree = shift ; my $GetDependenciesOnly = sub { my $tree = shift ; if('HASH' eq ref $tree) { return( 'HASH', undef, sort grep {! /^__/} keys %$tree) ; } #~ return (Data::TreeDumper::DefaultNodesToDisplay($tree)) ; } ; p_tree($tree, $GetDependenciesOnly, ' ') ; } push @EXPORT, 'p_d' ; *p_d = \&p_dependencies ; #---------------------------------------------------------------------- push @EXPORT, 'p_node' ; sub p_node { my $tree = shift ; my $node_regex = shift || '.*' ; my $package_regex = shift || '.*' ; unless(ref $tree eq 'HASH') { PrintDebug "p_node is expecting a reference to a node or a tree\n" ; return() ; } my $name = $tree->{__NAME} ; if($name =~ /^__/) { PrintDebug "PBS - DisplayNodeInformation Can't display info about private node $name\n" ; PrintDebug "PBS - Which node would you like to see?\n" ; NodeChoice($tree, $node_regex) ; } else { local $tree->{__PBS_CONFIG}{DISPLAY_NODE_ORIGIN} = 1 ; local $tree->{__PBS_CONFIG}{DISPLAY_NODE_DEPENDENCIES} = 1 ; local $tree->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_CAUSE} = 1 ; local $tree->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_RULES} = 1 ; local $tree->{__PBS_CONFIG}{DISPLAY_NODE_CONFIG} = 1 ; local $tree->{__PBS_CONFIG}{DISPLAY_NODE_BUILDER} = 1 ; local $tree->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_POST_BUILD_COMMANDS} = 1 ; PBS::Information::DisplayNodeInformation($tree, $tree->{__PBS_CONFIG}) ; } } push @EXPORT, 'p_n' ; *p_n = \&p_node ; #---------------------------------------------------------------------- sub NodeChoice { my $tree = shift ; my $node_regex = shift || '.*' ; my %nodes ; my $index = 0 ; for(sort keys %{$tree}) { next if /^__/ ; next unless /$node_regex/ ; PrintDebug "$index/ '$_'\n" ; $nodes{$index} = $tree->{$_} ; $index++ ; } if(0 == $index) { PrintDebug "No node to display.\n" ; } if(1 == $index) { local $nodes{0}->{__PBS_CONFIG}{DISPLAY_NODE_ORIGIN} = 1 ; local $nodes{0}->{__PBS_CONFIG}{DISPLAY_NODE_DEPENDENCIES} = 1 ; local $nodes{0}->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_CAUSE} = 1 ; local $nodes{0}->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_RULES} = 1 ; local $nodes{0}->{__PBS_CONFIG}{DISPLAY_NODE_CONFIG} = 1 ; local $nodes{0}->{__PBS_CONFIG}{DISPLAY_NODE_BUILDER} = 1 ; local $nodes{0}->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_POST_BUILD_COMMANDS} = 1 ; PBS::Information::DisplayNodeInformation($nodes{0}, $nodes{0}->{__PBS_CONFIG}) ; return() ; } while(1) { PrintDebug "PBS> " ; my $user_choice = ; chomp $user_choice; return if $user_choice eq '' ; if(exists $nodes{$user_choice}) { local $nodes{$user_choice}->{__PBS_CONFIG}{DISPLAY_NODE_ORIGIN} = 1 ; local $nodes{$user_choice}->{__PBS_CONFIG}{DISPLAY_NODE_DEPENDENCIES} = 1 ; local $nodes{$user_choice}->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_CAUSE} = 1 ; local $nodes{$user_choice}->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_RULES} = 1 ; local $nodes{$user_choice}->{__PBS_CONFIG}{DISPLAY_NODE_CONFIG} = 1 ; local $nodes{$user_choice}->{__PBS_CONFIG}{DISPLAY_NODE_BUILDER} = 1 ; local $nodes{$user_choice}->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_POST_BUILD_COMMANDS} = 1 ; PBS::Information::DisplayNodeInformation($nodes{$user_choice}, $nodes{$user_choice}->{__PBS_CONFIG}) ; } } } #---------------------------------------------------------------------- my %breakpoints ; sub AddBreakpoint { my $breakpoint_name = shift ; PrintWarning("Redefining breakpoint '$breakpoint_name'.\n") if(exists $breakpoints{$breakpoint_name}) ; $breakpoints{$breakpoint_name} = {@_} ; } #---------------------------------------------------------------------- sub RemoveBreakpoints { my $breakpoint_regex = shift || '.*'; for my $breakpoint_name (sort keys %breakpoints) { next unless $breakpoint_name =~ $breakpoint_regex ; PrintDebug("Breakpoint '$breakpoint_name' removed. \n") ; delete $breakpoints{$breakpoint_name} ; } } #---------------------------------------------------------------------- sub ListBreakpoints { my $breakpoint_regex = shift || '.*'; for my $breakpoint_name (sort keys %breakpoints) { next unless $breakpoint_name =~ $breakpoint_regex ; PrintDebug(DumpTree($breakpoints{$breakpoint_name}, "$breakpoint_name:")) ; } } #---------------------------------------------------------------------- sub ActivateBreakpoints { my @breakpoint_regexes = @_ ; for my $breakpoint_name (sort keys %breakpoints) { for my $breakpoint_regex (@breakpoint_regexes) { next unless $breakpoint_name =~ $breakpoint_regex ; $breakpoints{$breakpoint_name}{ACTIVE} = 1 ; PrintDebug("Breakpoint '$breakpoint_name' activated. \n") ; } } } #---------------------------------------------------------------------- sub DeactivateBreakpoints { my @breakpoint_regexes = @_ ; for my $breakpoint_name (sort keys %breakpoints) { for my $breakpoint_regex (@breakpoint_regexes) { next unless $breakpoint_name =~ $breakpoint_regex ; $breakpoints{$breakpoint_name}{ACTIVE} = 0 ; PrintDebug("Breakpoint '$breakpoint_name' deactivated. \n") ; } } } #---------------------------------------------------------------------- sub ActivatePerlDebugger { my @breakpoint_regexes = @_ ; for my $breakpoint_name (sort keys %breakpoints) { for my $breakpoint_regex (@breakpoint_regexes) { next unless $breakpoint_name =~ $breakpoint_regex ; $breakpoints{$breakpoint_name}{USE_DEBUGGER} = 1 ; PrintDebug("Breakpoint '$breakpoint_name' will activate the perl debugger. \n") ; } } } #---------------------------------------------------------------------- sub DeactivatePerlDebugger { my @breakpoint_regexes = @_ ; for my $breakpoint_name (sort keys %breakpoints) { for my $breakpoint_regex (@breakpoint_regexes) { next unless $breakpoint_name =~ $breakpoint_regex ; $breakpoints{$breakpoint_name}{USE_DEBUGGER} = 0 ; PrintDebug("Breakpoint '$breakpoint_name' will NOT activate the perl debugger. \n") ; } } } #---------------------------------------------------------------------- sub CheckBreakpoint { my %point = @_ ; return(0) unless $debug_enabled ; my $use_debugger = 0 ; for my $breakpoint_name (keys %breakpoints) { my $breakpoint = $breakpoints{$breakpoint_name} ; next unless $breakpoint->{ACTIVE} ; if(uc $breakpoint->{TYPE} eq 'BUILD') { next unless $point{TYPE} eq 'BUILD' ; } if(uc $breakpoint->{TYPE} eq 'POST_BUILD') { next unless $point{TYPE} eq 'POST_BUILD' ; } if(uc $breakpoint->{TYPE} eq 'TREE') { next unless $point{TYPE} eq 'TREE' ; } if(uc $breakpoint->{TYPE} eq 'INSERT') { next unless $point{TYPE} eq 'INSERT' ; } if(uc $breakpoint->{TYPE} eq 'VARIABLE') { next unless $point{TYPE} eq 'VARIABLE' ; } if(uc $breakpoint->{TYPE} eq 'DEPEND') { next unless $point{TYPE} eq 'DEPEND' ; } if(defined $breakpoint->{RULE_REGEX}) { next unless $point{RULE_NAME} =~ /$breakpoint->{RULE_REGEX}/ ; } if(defined $breakpoint->{NODE_REGEX}) { next unless $point{NODE_NAME} =~ /$breakpoint->{NODE_REGEX}/ ; } if(defined $breakpoint->{PACKAGE_REGEX}) { next unless $point{PACKAGE_NAME} =~ /$breakpoint->{PACKAGE_REGEX}/ ; } if(defined $breakpoint->{PBSFILE_REGEX}) { next unless $point{PBSFILE} =~ /$breakpoint->{PBSFILE_REGEX}/ ; } if($point{PRE}) { next unless $breakpoint->{PRE} ; } if($point{POST}) { next unless $breakpoint->{POST} ; } if($breakpoint->{TRIGGERED}) { next unless $point{TRIGGERED} ; } $use_debugger++ if $breakpoint->{USE_DEBUGGER} ; for my $action (@{$breakpoint->{ACTIONS}}) { $action->(%point, BREAKPOINT_NAME => $breakpoint_name) ; } if($breakpoint->{USE_DEBUGGER}) { $use_debugger++ ; } } return($use_debugger) ; } #---------------------------------------------------------------------- 1 ; __END__ =head1 NAME PBS::Debug - debugger support for PBS =head1 SYNOPSIS use PBS::Debug ; AddBreakpoint ( 'hi' , DEPEND => 1 , PRE => 1 , ACTIONS => [ sub { PrintDebug "Hi there.\n" ; } ] ) ; ActivateBreakpoints('hi') ; =head1 DESCRIPTION This module defines subs that manipulate PBS breakpoints (explained in B reference manual). sub PrintBanner: display a friendly message when run under the perl debugger sub EnableDebugger sub p_h: display a help within the perl debugger sub p_b_wizard: not implemented sub p_list sub p_tree pretty print a tree sub p_dependencies pretty print the dependencies of a node sub p_node pretty prints a node sub AddBreakpoint add a PBS breakpoint sub RemoveBreakpoints remove one or more PBS breakpoint according to the name regex passed as argument sub ListBreakpoints list all the breakpoints defined within PBS sub ActivateBreakpoints activates one or more PBS breakpoints sub DeactivateBreakpoints does the opposite of the above sub ActivatePerlDebugger activates wether a breakpoint (or breapoints) jumps to the perl debbugger sub DeactivatePerlDebugger does the opposite of the above sub CheckBreakpoint =head2 Aliases p_list_breakpoints or p_L = ListBreakpoints p_activate_breakpoints or p_ab = ActivateBreakpoints p_deactivate_breakpoints or p_db = DeactivateBreakpoints p_jump_in_debugger or p_j = ActivatePerlDebugger p_B = RemoveBreakpoints =head2 EXPORT EnableDebugger AddBreakpoint RemoveBreakpoints ListBreakpoints ActivateBreakpoints DeactivateBreakpoints ActivatePerlDebugger DeactivatePerlDebugger CheckBreakpoint and their aliases =head1 BREAKPOINTs AddBreakpoint ( 'hi' , DEPEND => 1 , PRE => 1 , ACTIONS => [ sub { PrintDebug "Hi there.\n" ; } ] ) ; =head2 Breakpoint position Some breakpoints cab be triggered before or after (or both) something happends in the system. if PRE is set, the breakpoints triggers before the action takes place. The breakpoint is triggered after the action takes place if after the action if POST is set. This allows you to take a snapshot before something happends and compare after it has happened. =head2 Types B (PRE/POST): when building a node. B (PRE/POST): when a post build action is to be performed B: when a tree (or sub tree) is finished depending B When a node is inserted in the dependency tree. B: when a variable is set. B (PRE/POST) when depending a node. B can be used to trigger a breakpoint only when a rule has matched the node. =head2 Breakpoint filtering You can set any of the following to a qr// or string. Only actions matching all the regexes you set will trigger a breakpoint. =over 2 =item * RULE_REGEX =item * NODE_REGEX =item * PACKAGE_REGEX =item * PBSFILE_REGEX =back =head2 Actions =head3 ACTIONS B is an array reference containing sub references. All the subs are run. All debugging functionality (ie activating or adding breakpoints) are available within the subs. =head3 USE_DEBUGGER B if running under the perl debugger and B is set, PBS will jump into the debugger after the breakpoint. =head1 AUTHOR Khemir Nadim ibn Hamouda. nadim@khemir.net =head1 SEE ALSO B reference manual / debug section. =cut