BEGIN {
$Sub::Spec::Runner::VERSION = '0.11';
}
# ABSTRACT: Run a subroutine
use 5.010;
use strict;
use Log::Any '$log';
use Moo;
use Sub::Spec::Utils; # temp, for _parse_schema
# {SUBNAME => {done=>BOOL, spec=>SPEC, fldeps=>FLATTENED_DEPS, ...}, ...}
has _sub_data => (is => 'rw', default=>sub{{}});
# [SUBNAME, ...], the list of subroutine names in the order of execution
has _sub_list => (is => 'rw', default=>sub{[]});
# for subroutines to share data
has _stash => (is => 'rw', default=>sub{{}});
# index to _sub_list, the current running subroutine
has _i => (is => 'rw');
# shorter way to insert custom code rather than subclassing
has _pre_run => (is => 'rw');
has _post_run => (is => 'rw');
has _pre_sub => (is => 'rw');
has _post_sub => (is => 'rw');
has common_args => (is => 'rw');
has load_modules => (is => 'rw', default=>sub{1});
has stop_on_sub_errors => (is => 'rw', default=>sub{1});
has order_before_run => (is => 'rw', default=>sub{1});
has undo => (is => 'rw');
has dry_run => (is => 'rw', default=>sub{0});
has state => (is => 'rw');
sub __parse_schema {
Sub::Spec::Utils::_parse_schema(@_);
}
# flatten deps clauses by flattening 'all' clauses, effectively change this
# deps clause:
#
# {
# sub => 's1',
# all => [
# {sub => 's1'},
# {deb => {d1=>0, d2=>'>= 0', d3=>'= 1'},
# {and => [{sub => 's2'}, {sub=>'s3'}, {deb=>{d3=>0}}],
# ],
# },
#
# into:
#
# {
# sub => ['s1', 's2', 's3'], # also uniquify sub names
# deb => [{d1],
# }
#
# dies if 'any' or 'none' clauses are encountered.
sub __flatten_deps {
my ($deps, $res) = @_;
$res //= {};
while (my ($k, $v) = each %$deps) {
if ($k =~ /^(any|none)$/) {
die "Can't handle '$_' deps clause";
} elsif ($k eq 'all') {
__flatten_deps($_, $res) for @$v;
} else {
$res->{$k} //= [];
next if $k eq 'sub' && $v ~~ @{ $res->{$k} };
push @{ $res->{$k} }, $v;
}
}
$res;
}
# add main:: if unqualified
sub __normalize_subname {
my ($subname) = @_;
$subname =~ /(.*)::(.+)/ or return "main::$subname";
(length($1) ? $1 : "main") . "::" . $2;
}
sub get_spec {
my ($self, $subname) = @_;
my ($module, $sub) = $subname =~ /(.+)::(.+)/;
no strict 'refs';
# assume module already loaded (by add())
my $ms = \%{"$module\::SPEC"};
$ms->{$sub};
}
our $current_runner;
sub add {
my ($self, $subname, $args) = @_;
$subname = __normalize_subname($subname);
return if exists $self->_sub_data->{$subname};
$log->tracef('-> add(%s)', $subname);
$self->_sub_data->{$subname} = undef; # to avoid deep recursion on circular
# load module
my ($module, $sub);
$subname =~ /(.+)::(.+)/;
($module, $sub) = ($1, $2);
if ($self->load_modules) {
my $modulep = $module;
$modulep =~ s!::!/!g; $modulep .= ".pm";
die "Cannot load module $module: $@\n"
unless eval { require $modulep };
}
my $spec = $self->get_spec($subname);
die "Can't find spec in \$$module\::SPEC{$sub}\n"
unless $spec;
my $fldeps = {};
if ($spec->{deps}) {
__flatten_deps($spec->{deps}, $fldeps);
$log->tracef("fldeps for $subname = %s", $fldeps);
$log->tracef("Checking dependencies ...");
local $current_runner = $self;
my $res = Sub::Spec::Clause::deps::check($spec->{deps});
die "Unmet dependencies for sub $subname: $res\n" if $res;
}
my $subdata = {
name => $subname,
spec => $spec,
fldeps => $fldeps,
args => $args,
};
$self->_sub_data->{$subname} = $subdata;
push @{ $self->_sub_list }, $subname;
$log->trace("<- add()");
}
sub order_by_dependencies {
$log->tracef("-> order_by_dependencies()");
my ($self) = @_;
my %deps;
while (my ($sn, $sd) = each %{$self->{_sub_data}}) {
$deps{$sn} //= [];
my $sub_deps = $sd->{fldeps}{run_sub};
push @{ $deps{$sn} }, @$sub_deps if $sub_deps;
}
my $ds = Algorithm::Dependency::Source::HoA->new(\%deps);
my $ado = Algorithm::Dependency::Ordered->new(
source => $ds,
selected => []
);
unless ($ado) {
$log->error("Failed to set up dependency algorithm");
return 0;
}
my $subs = $ado->schedule_all;
unless (ref($subs) eq 'ARRAY') {
$log->error("schedule_all() failed");
return 0;
}
if ($self->undo) {
$log->trace("Reversing order of subroutines because ".
"we are running in undo mode");
$subs = [reverse @$subs];
}
$self->_sub_list($subs);
1;
}
sub todo_subs {
my ($self) = @_;
[ grep {!$self->_sub_data->{done}} @{$self->_sub_list} ];
}
sub done_subs {
my ($self) = @_;
[ grep {$self->_sub_data->{done}} @{$self->_sub_list} ];
}
sub _log_running_sub {
my ($self, $subname) = @_;
$log->infof("Running %s ...", $self->format_subname($subname));
}
sub run {
my ($self) = @_;
$log->tracef("-> run()");
return [400, "No subroutines to run, please add() some first"]
unless @{ $self->_sub_list };
if (defined $self->undo) {
$log->trace("Checking undo/reverse feature of subroutines ...");
while (my ($sn, $sd) = each %{$self->{_sub_data}}) {
my $f = $sd->{spec}{features};
return [412, "Cannot run with undo: $sn doesn't support ".
"undo/reverse/pure"]
unless $f && ($f->{undo} || $f->{reverse} || $f->{pure});
}
unless ($self->state) {
$log->trace("Creating state object ...");
$self->state(Sub::Spec::Runner::State->new);
}
}
if ($self->dry_run) {
$log->trace("Checking dry_run feature of subroutines ...");
while (my ($sn, $sd) = each %{$self->{_sub_data}}) {
my $f = $sd->{spec}{features};
return [412, "Cannot run with dry_run: $sn doesn't support ".
"dry_run"]
unless $f && ($f->{dry_run} || $f->{pure});
}
}
if ($self->order_before_run) {
return [412, "Cannot resolve dependencies, please check for circulars"]
unless $self->order_by_dependencies;
}
my $hook_res;
eval { $hook_res = $self->pre_run };
return [412, "pre_run() died: $@"] if $@;
return [412, "pre_run() didn't return true"] unless $hook_res;
my $num_success_runs = 0;
my $num_failed_runs = 0;
my %success_subs;
my %failed_subs;
my $res;
my $use_last_res_status;
RUN:
while (1) {
$self->{_i} = -1;
my $some_not_done;
my $jumped;
while (1) {
$self->{_i}++ unless $jumped;
$jumped = 0;
last unless $self->{_i} < @{ $self->_sub_list };
my $subname = $self->_sub_list->[ $self->{_i} ];
my $sd = $self->_sub_data->{$subname};
next if $sd->{done};
$some_not_done++;
$self->_log_running_sub($subname);
my $orig_i = $self->{_i};
eval { $hook_res = $self->pre_sub($subname) };
if ($@) {
$res = [500, "pre_sub(%s) died: $@"];
$use_last_res_status++;
last RUN;
}
unless ($hook_res) {
$res = [500, "pre_sub(%s) didn't return true"];
$use_last_res_status++;
last RUN;
}
next if $sd->{done}; # pre_sub might skip this sub
$jumped = $orig_i != $self->{_i};
if ($jumped) {
last unless $self->{_i} < @{ $self->_sub_list };
}
$orig_i = $self->{_i};
$res = $self->_run_sub($subname, $sd->{args});
$sd->{res} = $res;
if ($self->success_res($res)) {
$num_success_runs++;
$success_subs{$subname}++;
delete ($failed_subs{$subname});
} else {
$num_failed_runs++;
$failed_subs{$subname}++;
delete ($success_subs{$subname});
if ($self->stop_on_sub_errors) {
$use_last_res_status++;
last RUN;
}
}
$self->done($subname, 1);
$jumped = $orig_i != $self->{_i};
if ($jumped) {
last unless $self->{_i} < @{ $self->_sub_list };
}
$orig_i = $self->{_i};
eval { $hook_res = $self->post_sub($subname) };
if ($@) {
$res = [500, "post_sub(%s) died: $@"];
$use_last_res_status++;
last RUN;
}
unless ($hook_res) {
$res = [500, "post_sub(%s) didn't return true"];
$use_last_res_status++;
last RUN;
}
$jumped = $orig_i != $self->{_i};
}
last unless $some_not_done;
}
eval { $hook_res = $self->post_run };
if ($@) {
$res = [500, "post_run() died: $@"];
$use_last_res_status = 1;
}
unless ($hook_res) {
$res = [500, "post_run() didn't return true"];
$use_last_res_status = 1;
}
my $num_subs = scalar(@{$self->_sub_list});
my $num_success_subs = scalar(keys %success_subs);
my $num_failed_subs = scalar(keys %failed_subs );
unless ($use_last_res_status) {
$res = [];
}
$res->[2] = {
num_success_runs => $num_success_runs,
num_failed_runs => $num_failed_runs,
num_runs => $num_success_runs+$num_failed_runs,
num_success_subs => $num_success_subs,
num_failed_subs => $num_failed_subs,
num_subs => $num_subs,
num_run_subs => $num_success_subs+$num_failed_subs,
num_skipped_subs => $num_subs - ($num_success_subs+$num_failed_subs),
};
unless ($use_last_res_status) {
if ($num_success_subs) {
if ($num_failed_subs) {
$res->[0] = 200;
$res->[1] = "Some failed";
} else {
$res->[0] = 200;
$res->[1] = "All succeeded";
}
} else {
$res->[0] = 500;
$res->[1] = "All failed";
}
}
$log->tracef("<- run(), res=%s", $res);
$res;
}
sub _run_sub {
my ($self, $subname, $args) = @_;
$log->tracef("-> _run_sub(%s)", $subname);
my $res;
eval {
my ($module, $sub) = $subname =~ /(.+)::(.+)/;
my $fref = \&{"$module\::$sub"};
unless ($fref) {
$res = [500, "No sub \&$subname defined"];
last;
}
my $sd = $self->_sub_data->{$subname};
my $spec = $sd->{spec};
my %all_args;
my $common_args = $self->common_args // {};
for (keys %$common_args) {
$all_args{$_} = $common_args->{$_} if !$sd->{spec}{args} ||
$sd->{spec}{args}{$_};
}
$args //= {};
for (keys %$args) {
$all_args{$_} = $args->{$_} if !$sd->{spec}{args} ||
$sd->{spec}{args}{$_};
}
if (defined $self->undo) {
if ($spec->{features}{pure}) {
# do nothing
} elsif ($spec->{features}{reverse}) {
$all_args{-reverse} = $self->undo;
} elsif ($spec->{features}{undo}) {
$all_args{-undo} = $self->undo;
$all_args{-state} = $self->state;
}
}
if (defined $self->dry_run) {
$all_args{-dry_run} = $self->dry_run;
}
$log->tracef("-> %s(%s)", $subname, \%all_args);
$all_args{-runner} = $self;
$res = $fref->(%all_args);
$log->tracef("<- %s(), res=%s", $subname, $res);
};
$res = [500, "sub died: $@"] if $@;
$log->tracef("<- _run_sub(%s), res=%s", $subname, $res);
$res;
}
sub format_subname {
$_[1];
}
sub success_res {
my ($self, $res) = @_;
$res->[0] >= 200 && $res->[0] <= 399;
}
sub pre_run {
my $self = shift;
!$self->_pre_run || $self->_pre_run->($self, @_);
}
sub pre_sub {
my $self = shift;
!$self->_pre_sub || $self->_pre_sub->($self, @_);
}
sub post_sub {
my $self = shift;
!$self->_post_sub || $self->_post_sub->($self, @_);
}
sub post_run {
my $self = shift;
!$self->_post_run || $self->_post_run->($self, @_);
}
sub result {
my ($self, $subname) = @_;
$subname = __normalize_subname($subname);
die "$subname is not in the list of added subroutines\n"
unless $self->_sub_data->{$subname};
$self->_sub_data->{$subname}{res};
}
sub done {
my ($self, $subname0, $newval) = @_;
my @subnames;
if (ref($subname0) eq 'Regexp') {
@subnames = grep {/$subname0/} @{$self->_sub_list};
} else {
push @subnames, __normalize_subname($subname0);
}
my $oldval;
for my $subname (@subnames) {
unless ($self->_sub_data->{$subname}) {
die "$subname is not in the list of added subroutines\n";
}
$oldval = $self->_sub_data->{$subname}{done};
if (defined($newval)) {
$self->_sub_data->{$subname}{done} = $newval;
}
}
$oldval;
}
sub skip {
my ($self, $subname) = @_;
$self->done($subname, 1);
}
sub skip_all {
my ($self) = @_;
$self->done(qr/.*/, 1);
}
sub repeat {
my ($self, $subname) = @_;
$self->done($subname, 0);
}
sub repeat_all {
my ($self) = @_;
$self->done(qr/.*/, 1);
}
# return subname and all others which directly/indirectly depends on it
sub _find_dependants {
my ($self, $subname, $res) = @_;
$res //= [$subname];
my $sd = $self->_sub_data;
for (keys %$sd) {
next if $_ ~~ @$res;
if ($subname ~~ @{ $sd->{$_}{fldeps}{run_sub} // [] }) {
push @$res, $_;
$self->_find_dependants($_, $res);
}
}
$res;
}
sub branch_done {
my ($self, $subname, $newval) = @_;
$subname = __normalize_subname($subname);
unless ($self->_sub_data->{$subname}) {
die "$subname is not in the list of added subroutines";
return;
}
my $sn = $self->_find_dependants($subname);
for (@$sn) {
$self->_sub_data->{$_}{done} = $newval;
}
}
sub jump {
my ($self, $subname) = @_;
$subname = __normalize_subname($subname);
unless ($self->_sub_data->{$subname}) {
die "$subname is not in the list of added subroutines\n";
}
my $sl = $self->_sub_list;
for my $i (0..@$sl-1) {
if ($sl->[$i] eq $subname) {
$self->_i($i);
last;
}
}
}
sub stash {
my ($self, $key, $newval) = @_;
my $oldval = $self->_stash->{$key};
if (defined($newval)) {
$self->_stash->{$key} = $newval;
}
$oldval;
}
BEGIN {
$Sub::Spec::Clause::deps::VERSION = '0.11';
}
# XXX adding run_sub should be done locally, and also modifies the spec schema
# (when it's already defined). probably use a utility function add_dep_clause().
sub check_run_sub {
my ($cval) = @_;
$Sub::Spec::Runner::current_runner->add($cval);
"";
}
1;
=pod
=head1 NAME
Sub::Spec::Runner - Run a subroutine
=head1 VERSION
version 0.11
=head1 SYNOPSIS
In YourModule.pm:
package YourModule;
use 5.010;
our %SPEC;
$SPEC{a} = { deps => {run_sub=>'b'}, ... };
sub a { my %args = @_; say "a"; [200, "OK"] }
$SPEC{b} = { deps => {run_sub=>'c'}, ... };
sub b { my %args = @_; say "b"; [200, "OK"] }
$SPEC{c} = { deps => {all=>[{run_sub=>'d'}, {run_sub=>'e'}]}, ... };
sub c { my %args = @_; say "c"; [200, "OK"] }
$SPEC{d} = { deps => {run_sub=>'e'}, ... };
sub d { my %args = @_; say "d"; [200, "OK"] }
$SPEC{e} = { ... };
sub e { my %args = @_; say "e"; [200, "OK"] }
In main module:
use Sub::Spec::Runner;
my $runner = Sub::Spec::Runner->new(load_modules=>0);
$runner->add('YourModule::a');
$runner->run;
Will output:
e
d
c
b
a
=head1 DESCRIPTION
This class "runs" a subroutine. "Running" basically means loading the module and
calling the subroutine, plus a few other stuffs. See run() for more details.
This module uses L<Log::Any> logging framework. Use something like
L<Log::Any::App>, etc to see more logging statements for debugging.
This module uses L<Moo> for object system.
=head1 ATTRIBUTES
=head2 common_args => HASHREF
Arguments to pass to each subroutine. Note that each argument will only be
passed if the 'args' clause in subroutine spec specifies that the sub accepts
that argument, or if subroutine doesn't have an 'args' clause. Example:
package Foo;
our %SPEC;
$SPEC{sub0} = {};
sub sub0 { ... }
$SPEC{sub1} = {args=>{}};
sub sub1 { ... }
$SPEC{sub2} = {args=>{foo=>"str"}};
sub sub2 { ... }
$SPEC{sub3} = {args=>{bar=>"str"}};
sub sub2 { ... }
$SPEC{sub4} = {args=>{foo=>"str", bar=>"str"}};
sub sub4 { ... }
package main;
use Sub::Spec::Runner;
my $runner = Sub::Spec::Runner->new(common_args => {foo=>1, foo=>2});
$runner->add("Foo::sub$_") for (1 2 3 4);
$runner->run;
Then only sub0 and sub4 will receive the 'foo' and 'bar' args. sub1 won't
receive any arguments, sub2 will only receive 'foo', sub3 will only receive
'bar'.
=head2 load_modules => BOOL (default 1)
Whether to load (require()) modules when required (in add()). You can turn this
off if you are (or want to make) sure that all the subroutines to be run are
already loaded.
=head2 stop_on_sub_errors => BOOL (default 1)
When run()-ning subroutines, whether a non-success return value from a
subroutine stops the whole run. If turned off, run() will continue to the next
subroutine. Note that you can override what constitutes a success return value
by overriding success_res().
=head2 order_before_run => BOOL (default 1)
Before run() runs the subroutines, it will call order_by_dependencies() to
reorder the added subroutines according to dependency tree (the 'sub_run'
dependency clause). You can turn off this behavior by setting this attribute to false.q
=head2 undo => BOOL (default undef)
If set to 0 or 1, then these things will happen: 1) Prior to running, all added
subroutines will be checked and must have 'undo' or 'reverse' feature, or are
'pure' (see L<Sub::Spec::Clause::features> for more details on specifying
features). 2) '-undo' and '-state' special argument will be given with value 0/1
to each sub supporting undo (or '-reverse' 0/1 for subroutines supporting
reverse). No special argument will be given for pure subroutines.
Additionally, if 'undo' is set to 1, then order_by_dependencies() will reverse
the order of run. This will only be done if 'order_before_run' attribute is set
to true. Otherwise, you might have to do the reversing of order by yourself, if
so desired.
In summary: setting to 0 means run normally, but instruct subroutines to store
undo information to enable undo in the future. Setting to 1 means undo. Setting
to undef (the default) means disregard undo stuffs.
=head2 dry_run => BOOL (default 0)
If set to 0 or 1, then these things will happen: 1) Prior to running, all added
subroutines will be checked and must have 'dry_run' feature, or are 'pure' (see
L<Sub::Spec::Clause::features> for more details on specifying features). 2)
'-dry_run' special argument will be given with value 0/1 to each sub supporting
dry run. No special argument will be given for pure subroutines.
=head2 state => STATE OBJECT
Used to store state object, set by create_state_obj() when running under undo
mode. State object is used by subroutines to store undo information.
=head1 METHODS
=head2 $runner->get_spec($subname) => SPEC
Get spec for sub named $subname. Will be called by add(). Can be overriden to
provide your own specs other than from %SPECS package variables.
=head2 $runner->add($subname[, $args])
Add subroutine to the set of subroutines to be run. Example:
$runner->add('Package::subname');
Will first get the sub spec by loading the module and read the %SPEC package var
(this behavior can be changed by overriding get_spec()). Will not load modules
if 'load_modules' attribute is false. Will die if cannot get spec.
After that, it will check dependencies (the 'deps' spec clause) and die if some
dependencies are unmet. All subroutine names mentioned in 'run_sub' dependency
clause will also be add()-ed automatically.
=head2 $runner->order_by_dependencies()
Reorder set of subroutines by dependencies. Normally need not be called manually
since it will be caled by run() prior to running subroutines, but if you turn
off 'order_before_run' attribute, you'll need to call this method explicitly if
you want ordering.
Will return a true value on success, or false if dependencies cannot be resolved
(e.g. there is circular dependency).
=head2 $runner->todo_subs() => ARRAYREF
Return the current list of subroutine names not yet runned, in order. Previously
run subroutines can belong to this list again if repeat()-ed.
=head2 $runner->done_subs() => ARRAYREF
Return the current list of subroutine names already run, in order. Never-run
subroutines can belong to this list too if skip()-ed.
=head2 $runner->run() => [STATUSCODE, ERRMSG, RESULT]
Run (call) a set of subroutines previously added by add().
First it will check 'undo' attribute. If defined, then all added subroutines are
required to have undo/reverse/pure feature or otherwise run() will immediately
return with error 412. After that state object (L<Sub::Spec::Runner::State>
instance) will be created and 'state' attribute will be set to this.
Sub::Spec::Runner::State is a state object which stores data in YAML files
under ~/.subspec/.undo/, one per subroutine. You can use your own state object
by setting the 'state' attribute before this. If 'state' attribute is already
set, run() will not overwrite it.
Then it will check 'dry_run' attribute. If true, then all added subroutines are
required to have dry_run/pure feature or otherwise run() will immediately return
with error 412.
After that it will call order_by_dependencies() to reorder the subroutines
according to dependency order. This can be turned off via setting
'order_before_run' attribute to false.
After that, it will call pre_run(), which you can override. pre_run() must
return true, or run() will immediately return with 412 error.
Then it will call each subroutine successively. Each subroutine will be called
with arguments specified in 'common_args' attribute and args specified in add(),
with one extra special argument, '-runner' which is the runner object. Prior to
running a subroutine, pre_sub() will be called. It must return true, or run()
will immediately return with 500 error.
Runner will store the return value of each subroutine. Exception from subroutine
will be trapped by eval() and upon exception return value of subroutine is
assumed to be 500.
The subroutine being run can see the status/result of other subroutines by
calling $runner->done($subname), $runner->result($subname). It can share data by
using $runner->stash(). It can also repeat/skip some subroutines by calling
$runner->skip(), skip_all(), repeat(), repeat_all(), branch_done(). It can jump
to other subroutines using $runner->jump(). See the respective method
documentation for more details.
After running a subroutine, post_sub() will be called. It must return true, or
run() will immediately return with 500 error.
If 'stop_on_sub_errors' attribute is set to true (the default), then if the
subroutine returns a non-success result, run() will immediately exit with that
result. The meaning of subroutine's success can be changed by overriding
success_res() (by default, all 2xx and 3xx are considered success).
After all subroutines have been run (or skipped), run() will call post_run()
which must return true. Otherwise run() will immediately exit with 500 status.
After that, run() will return the summary in RESULT (number of subroutines run,
skipped, successful, etc). It will return status 200 if there are at least one
subroutine returning success, or 500 otherwise.
=head2 $runner->format_subname($subname) => STR
Can be used to format info log message: "Running XXX ..." when about to run a
subroutine inside run(). Default is "Running Package::bar ..." (just the
subname)
=head2 $runner->success_res($res) => BOOL
By default, all responses with 2xx and 3xx are assumed as a success. You can
override this.
=head2 $runner->pre_run() => BOOL
See run() for more details. Can be overridden by subclass.
=head2 $runner->pre_sub() => BOOL
See run() for more details. Can be overridden by subclass.
=head2 $runner->post_sub() => BOOL
See run() for more details. Can be overridden by subclass.
=head2 $runner->post_run() => BOOL
See run() for more details. Can be overridden by subclass.
=head2 $runner->result(SUBNAME) => RESULT
Return the result of run subroutine named SUBNAME. If subroutine is not run yet,
will return undef. Will die if subroutine is not in the list of added
subroutines.
=head2 $runner->done(SUBNAME[, VALUE]) => OLDVAL
If VALUE is set, set a subroutine to be done/not done. Otherwise will return the
current done status of SUBNAME. Will die if subroutine named SUBNAME is not in
the list of added subs.
SUBNAME can also be a regex, which means all subroutines matching the regex. The
last SUBNAME's current done status will be returned.
=head2 $runner->skip(SUBNAME)
Alias for done(SUBNAME, 1).
=head2 $runner->skip_all()
Alias for skip(qr/.*/, 1).
=head2 $runner->repeat(SUBNAME)
Alias for done(SUBNAME, 0).
=head2 $runner->repeat_all()
Alias for repeat(qr/.*/, 1).
=head2 $runner->branch_done(SUBNAME, VALUE)
Just like done(), except that will set SUBNAME *and all its dependants*.
Example: if a depends on b and b depends on c, then doing branch_done(c, 1) will
also set a & b as done.
SUBNAME must be a string and not regex.
=head2 $runner->jump($subname)
Jump to another subname. Can be called in pre_sub() or inside subroutine or
post_sub().
=head2 $runner->stash(NAME[, VALUE]) => OLDVAL
Get/set stash data. This is a generic place to share data between subroutines
being run.
=head1 SEE ALSO
L<Sub::Spec>
=head1 AUTHOR
Steven Haryanto <stevenharyanto@gmail.com>
=head1 COPYRIGHT AND LICENSE
This software is copyright (c) 2011 by Steven Haryanto.
This is free software; you can redistribute it and/or modify it under
the same terms as the Perl 5 programming language system itself.
=cut
__END__