NAME
Fennec - A test helper providing RSPEC, Workflows, Parallelization, and Encapsulation.
DESCRIPTION
Fennec started as a project to improve the state of testing in Perl. Fennec looks to existing solutions for most problems, so long as the existing solutions help meet the features listed below.
API STABILITY
Fennec versions below 1.000 were considered experimental, and the API was subject to change. As of version 1.0 the API is considered stabalized. New versions may add functionality, but not remove or significantly alter existing functionality.
FEATURES
- Forking Works
-
Forking in tests just plain works. You can fork, and run assertions (tests) in both processes.
- Test groups can be run alone
-
Encapsulated test groups can be run individually, without running the entire file. (See Test::Workflow)
- Parallelization within test files
-
Encapsulated test groups can be run in parallel if desired. (On by default with up to 3 processes)
See the "PARALLEL" for details about what runs in what processes.
- Test reordering
-
Tests groups can be sorted, randomized, or sorted via a custom method. (see Test::Workflow)
- Test::Builder and Test::Builder2 compatibility
-
Fennec is compatible with Test::Builder based tools. Test::Builder2 support is in-place, but experimental until Test::Builder2 is officially released.
- Ability to decouple from Test::Builder
-
Fennec is configurable to work on alternatives to Test::Builder.
- No need to formally end tests
-
You do not need to put anything such as done_testing() at the end of your test file.
- Test counting is handled for you
-
You do not need to worry about test counts.
- Diagnostic messages are grouped with the failed test
-
Annoyed when your test failure and the diagnostics messages about that test are decoupled?
ok 1 - foo ok 2 - bar not ok 3 - baz ok 4 - bannana ok 5 - pear # Test failure on line 67 # expected: 'baz' # got: 'bazz'
This happens because normal output is sent to STDOUT, while errors are sent to STDERR. This is important in a non-verbose harness so that you can still see error messages. In a verbose harness however it is just plain annoying. Fennec checks the verbosity of the harness, and sends diagnostic messages to STDOUT when the harness is verbose.
Note: This is not IO redirection or handle manipulation, your warnings and errors will still go to STDERR.
SYNOPSIS
package MyTest;
use strict;
use warnings;
use Fennec;
tests foo => sub {
ok( 1, 'bar' );
};
tests another => sub {
ok( 1, 'something passed' );
};
tests not_ready => (
todo => "Feature not implemented",
code => sub { ... },
);
tests very_not_ready => (
skip => "These tests will die if run"
code => sub { ... },
);
1;
By default these test groups will be run in parallel. They will also be run in random order by default. See the "CONFIGURATION" for more details on controlling behavior. Also see Test::Workflow for more useful and poweful test groups and structures.
FRIENDLIER INTERFACE
If you use Fennec::Declare you can write tests like this:
package MyTest;
use strict;
use warnings;
use Fennec;
tests foo {
ok( 1, 'bar' );
}
1;
Thats right, no => sub
and no trailing ';'.
RUNNING ONLY A SPECIFIC GROUP
1: package MyTest;
2: use strict;
3: use warnings;
4: use Fennec;
5:
6: tests foo => sub {
7: ok( 1, 'bar' );
8: };
9:
10: tests another => sub {
11: ok( 1, 'something passed' );
12: };
13:
14: 1;
In the above code there are 2 test groups, 'foo', and 'another'. If you wanted, you could run just one, without the others running. Fennec looks at the 'FENNEC_TEST' environment variable. If the variable is set to a string, then only the test groups with that string as a name will run.
$ FENNEC_TEST="foo" prove -Ilib -v t/FennecTest.t
In addition, you could provide a line number, and only the test group defined across that line will be run. For example, to run 'foo' you could give the line number 6, 7 or 8 to run that group alone.
$ FENNEC_TEST="7" prove -Ilib -v t/FennecTest.t
This will run only test 'foo'. The use of line numbers makes editor integration very easy. Most editors will let you bind a key to running the above command replacing t/FennecTest.t with the current file, and automatically inserting the current line into FENNEC_TEST.
EDITOR INTEGRATION
VI/VIM
Insert this into your .vimrc file to bind the F8 key to running the current test in the current file:
function! RunFennecLine()
let cur_line = line(".")
exe "!FENNEC_TEST='" . cur_line . "' prove -v -I lib %"
endfunction
" Go to command mode, save the file, run the current test
:map <F8> <ESC>:w<cr>:call RunFennecLine()<cr>
:imap <F8> <ESC>:w<cr>:call RunFennecLine()<cr>
MODULES LOADED AUTOMATICALLY WITH FENNEC
- Test::More
-
The standard perl test library.
- Test::Exception
-
One of the more useful test libraries, used to test code that throws exceptions (dies).
- Test::Warn
-
Test code that issues warnings.
- Test::Workflow
-
Provides RSPEC, and several other workflow related helpers. Also provides the test group encapsulation.
- Mock::Quick
-
Quick and effective mocking with no action at a distance side effects.
MODULES FENNEC MAKES AN EFFORT TO SUPPORT
- Test::Class
-
A Fennec class can also be a Test::Class class.
- Test::Builder
-
If Fennec did not support this who would use it?
- Test::Builder2
-
There is currently experimental support for Test::Builder2. Once Test::Builder2 is officially released, support will be finalized.
PARALLEL
When tests run in parallel (default) the following notes should be observed.
- parallel => 1 means fork for test blocks, but only run 1 proc at a time.
- parallel => 0 turns forking off completely
- describe and cases run in the parent process
-
describe something => sub { ... }
Blocks like the above run in the parent process, all will run BEFORE any test or state-building blocks.
- test blocks run in their own processes
-
tests foo => sub { ... };
Test blocks like this will all run in their own process.
- before_each, after_each and around_each run in the test block process
-
before_each set_it => sub { ... }; tests test_it => sub { ... };
XXX_each
will run in the same process as the test block, that is after the fork() call. - before_all, after_all, and around_all run in the parent process
-
before_all set_once => sub { ... }; tests test_it => sub { ... };
XXX_all
will run in the parent process, that is before fork() is called to run the test block. - each case + test combination runs in its own process
-
case c1 { ... } case c2 { ... } test t1 { ... } test t2 { ... }
This effectively builds 4 combinations to run: c1+t1, c1+t2, c2+t1, c2+t2. Each of these 4 combinations will be run in their own process. The case will run first, followed by the test block.
Be aware, it is easy to define an exponential number of tests using the case+test combiner.
CONFIGURATION
There are 2 ways to configure Fennec. One is to specify configuration options at import. The other is to subclass Fennec and override the defaults() method.
Configuration options:
utils => [ qw/ModuleA ModuleB .../ ]
Provide a list of modules to load. They will be imported as if you typed use MODULE
.
You can specify arguments for each class like so:
use Fennec utils => [ 'My::Util' ],
'My::Util' => [ 'Arg1', 'Arg2' ];
parallel => $MAX
Specify the maximum number of processes Fennec should use to run your tests. Set to 0 to never create a new process. Depedning on conditions 1 MAY fork for test groups while still only running 1 at a time, but this behavior is not guarenteed.
Default: 3
runner_class => $CLASS
Specify the runner class. Default: Fennec::Runner
with_tests => \@CLASSES
Load test_groups and workflows from another class. This allows you to put test groups common to many test files into a single place for re-use.
test_sort => $SORT
This sets the test sorting method for Test::Workflow test groups. Accepts 'random', 'sort', a codeblock, or 'ordered'. This uses a fuzzy matching, you can use the shorter versions 'rand', and 'ord'.
Defaults to: 'rand'
debug_long_running => $TIMEOUT
This will cause a test block to abort after a specified timeout (value is passed directly to alaram).
NOTE This uses the alarm($timeout)
function. If your tests include alarms the behavior is not defined. One will certainly clobber the other, your will most likely come out on top, but that is not guarenteed in any way. Only use this while debugging and remove it afterwords.
- 'random'
-
Will shuffle the order. Keep in mind Fennec sets the random seed using the date so that tests will be determinate on the day you write them, but random over time.
- 'sort'
-
Sort the test groups by name. When multiple tests are wrapped in before_all or after_all the describe/cases block name will be used.
- 'ordered'
-
Use the order in which the test groups were defined.
- sub { my @tests = @_; ...; return @new_tests }
-
Specify a custom method of sorting. This is not the typical sort {} block, $a and $b will not be set.
AT IMPORT
use Fennec parallel => 5,
utils => [ 'My::Util' ],
... Other Options ...;
BY SUBCLASS
package My::Fennec;
use base 'Fennec';
sub defaults {(
utils => [qw/
Test::More Test::Warn Test::Exception Test::Workflow
/],
utils_with_args => {
My::Util => [qw/function_x function_y/],
},
parallel => 5,
runner_class => 'Fennec::Runner',
)}
# Hook, called after import
sub init {
my $class = shift;
# All parameters passed to import(), as well as caller => [...] and meta => $meta
my %params = @_;
...
}
1;
MORE COMPLETE EXAMPLE
This is a more complete example than that which is given in the synopsis. Most of this actually comes from Method::Workflow, See those docs for more details. Significant sections are in separate headers, but all examples should be considered part of the same long test file.
NOTE: All blocks, including setup/teardown are methods, you can shift @_ to get $self.
BASIC EXAMPLES
package MyTest;
use strict;
use warnings;
use Fennec parallel => 2,
with_tests => [qw/ Test::TemplateA Test::TemplateB /],
test_sort => 'rand';
# Tests can be at the package level
use_ok( 'MyClass' );
# Fennec works with Test::Class
use base 'Test::Class';
sub tc_test : Test(1) {
my $self = shift;
ok( 1, 'This is a Test::Class test' );
}
tests loner => sub {
my $self = shift;
ok( 1, "1 is the loneliest number... " );
};
tests not_ready => (
todo => "Feature not implemented",
code => sub { ... },
);
tests very_not_ready => (
skip => "These tests will die if run"
code => sub { ... },
);
RSPEC WORKFLOW
Here setup/teardown methods are declared in the order in which they are run, but they can really be declared anywhere within the describe block and the behavior will be identical.
describe example => sub {
my $self = shift;
my $number = 0;
my $letter = 'A';
before_all setup => sub { $number = 1 };
before_each letter_up => sub { $letter++ };
# it() is an alias for tests()
it check => sub {
my $self = shift;
is( $letter, 'B', "Letter was incremented" );
is( $number, 2, "number was incremented" );
};
after_each reset => sub { $number = 1 };
after_all teardown => sub {
is( $number, 1, "number is back to 1" );
};
describe nested => sub {
# This nested describe block will inherit before_each and
# after_each from the parent block.
...
};
describe maybe_later => (
todo => "We might get to this",
code => { ... },
);
};
FENNEC'S RSPEC IMPROVEMENT
Fennec add's to the RSPEC toolset with the around keyword.
describe addon => sub {
my $self = shift;
around_each localize_env => sub {
my $self = shift;
my ( $inner ) = @_;
local %ENV = ( %ENV, foo => 'bar' );
$inner->();
};
tests foo => sub {
is( $ENV{foo}, 'bar', "in the localized environment" );
};
};
CASE WORKFLOW
Cases are used when you have a test that you wish to run under several r tests conditions. The following is a trivial example. Each test will be run once under each case. Beware! this will run (cases x tests), with many tests and cases this can be a huge set of actual tests. In this example 8 in total will be run.
Note: The 'cases' keyword is an alias to describe. case blocks can go into any workflow and will work as expected.
cases check_several_numbers => sub {
my $number;
case one => sub { $number = 2 };
case one => sub { $number = 4 };
case one => sub { $number = 6 };
case one => sub { $number = 8 };
tests is_even => sub {
ok( !$number % 2, "number is even" );
};
tests only_digits => sub {
like( $number, qr/^\d+$/i, "number is all digits" );
};
};
1;
MOCKING FROM MOCK::QUICK
Mock::Quick is imported by default. Mock::Quick is a powerful mocking library with a very friendly syntax.
MOCKING OBJECTS
use Mock::Quick;
my $obj = obj(
foo => 'bar', # define attribute
do_it => qmeth { ... }, # define method
...
);
is( $obj->foo, 'bar' );
$obj->foo( 'baz' );
is( $obj->foo, 'baz' );
$obj->do_it();
# define the new attribute automatically
$obj->bar( 'xxx' );
# define a new method on the fly
$obj->baz( qmeth { ... });
# remove an attribute or method
$obj->baz( qclear() );
MOCKING CLASSES
use Mock::Quick;
my $control = qclass(
# Insert a generic new() method (blessed hash)
-with_new => 1,
# Inheritance
-subclass => 'Some::Class',
# Can also do
-subclass => [ 'Class::A', 'Class::B' ],
# generic get/set attribute methods.
-attributes => [ qw/a b c d/ ],
# Method that simply returns a value.
simple => 'value',
# Custom method.
method => sub { ... },
);
my $obj = $control->packahe->new;
# Override a method
$control->override( foo => sub { ... });
# Restore it to the original
$control->restore( 'foo' );
# Remove the anonymous namespace we created.
$control->undefine();
TAKING OVER EXISTING CLASSES
use Mock::Quick;
my $control = qtakeover( 'Some::Package' );
# Override a method
$control->override( foo => sub { ... });
# Restore it to the original
$control->restore( 'foo' );
# Destroy the control object and completely restore the original class Some::Package.
$control = undef;
MOCKING EXPORTS
Mock-Quick uses Exporter::Declare. This allows for exports to be prefixed or renamed. See "RENAMING IMPORTED ITEMS" in Exporter::Declare for more information.
- $obj = qobj( attribute => value, ... )
-
Create an object. Every possible attribute works fine as a get/set accessor. You can define other methods using qmeth {...} and assigning that to an attribute. You can clear a method using qclear() as an argument.
See Mock::Quick::Object for more.
- $control = qclass( -config => ..., name => $value || sub { ... }, ... )
-
Define an anonymous package with the desired methods and specifications.
See Mock::Quick::Class for more.
- $control = qtakeover( $package )
-
Take control over an existing class.
See Mock::Quick::Class for more.
- qclear()
-
Returns a special reference that when used as an argument, will cause Mock::Quick::Object methods to be cleared.
- qmeth { my $self = shift; ... }
-
Define a method for an Mock::Quick::Object instance.
ADDITIONAL USER DOCUMENTATION
SEE ALSO
- Fennec::Lite
- Test::Workflow
- Fennec::Runner
- Mock::Quick
- Test::More
- Test::Exception
- Test::Warn
- Test::Class
- Test::Builder
NOTES
When you use Fennec
, it will check to see if you called the file directly. If you directly called the file Fennec will restart Perl and run your test through Fennec::Runner.
CAVEATS
When running a test group by line, Fennec takes it's best guess at which group the line number represents. There are 2 ways to get the line number of a codeblock:
The first is to use the B module. The B module will return the line of the first statement within the codeblock.
The other is to define the codeblock in a function call, such as tests foo => sub {...}
, tests() can then use caller() which will return the last line of the statement.
Combining these methods, we can get the approximate starting and ending lines for codeblocks defined through Fennec's keywords.
This will break if you do something like:
tests foo => \&my_test;
sub my_test { ... }
But might work just fine if you do:
tests foo => \&my_test;
sub my_test { ... }
But might run both tests in this case when asking to run 'baz' by line number:
tests foo => \&my_test;
tests baz => sub {... }
sub my_test { ... }
AUTHORS
Chad Granum exodist7@gmail.com
COPYRIGHT
Copyright (C) 2011 Chad Granum
Fennec is free software; Standard perl licence.
Fennec is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the license for more details.