NAME

Test::Suite::Specification - Specification for Test::Suite

DESCRIPTION

Test-Suite is a test module that addresses several complains I have heard, or have myself issued forth about perl testing.

TOOLS

$ prove_suite file(s)
$ prove_suite file1 [case(s)] [set(s)] file2 ...
$ prove_suite dir

Command line tool to run specified tests.

Test::Suite::Tester->run( @PARAMS )

Used to run Test::Suite tests from a test script.

SYNOPSYS

package MyModule;
use strict;
use warnings;

package MyModuleTest;
use strict;
use warnings;

# This is also equivilent to 'use MyModule @args;'
use Test::Suite testing => 'MyModule', import_args => \@args,
    # Load specified Test::Suite::XXX plugins, 'more', 'exception', and
    # 'warn' are automatically loaded unless specified with a '-' prefix.
    plugins => [qw//],
    # if 'case' then run all cases in parrallel (but sets in sequence)
    # if 'set' then run all cases in sequence, but sets in parrallel within them
    parallel => BOOL || 'case' || 'set'
    # Default is true, to randomize order of cases/sets
    random => BOOL,
    # Specify custom set/case default defenition options
    case_defaults => { ... },
    set_defaults => { ... },
;

# Create the optional initialize method that will take care of setup tasks
# that occur before any case or set is run. Will only be run automatically
# once.
sub initialize {
    # Will be a 'MyModuleTest' object with all methods you define in the
    # file, as well as accessors to find current set and case.
    my $self = shift;
}

# Define test set 'a', and a method for it to run
test_set a => (
    # make this whole set todo
    todo => 'reason',
    # skip this whole set
    skip => 'reason',
    # Optional, assert that the set runs specified number of tests
    tests => COUNT,
    # If true than this set will only run in the main process, not a fork.
    # This will prevent it from being run in parallel with others.
    # Default: false
    no_fork => BOOL,
    # force the set to be run in a forked process, useful for tests that
    # might mangle things horribly for other tests.
    force_fork => BOOL,
    # Specify the method to run for this set, defaults to 'set_NAME'
    method => 'name' || \&code,
    # What to do if there is a test failure
    on_fail => OPTION,
        'finish'      # Just keep going if possible, otherwise next set.
        'next_set'    # Move to the next set.
        'next_case'   # Move to the next case.
        'next_module' # Move to the next test module.
        'abort'       # Stop all testing and exit.
);
# Define the test code automatically used by set 'a'
sub set_a {
    # Will be a 'MyModuleTest' object with all methods you define in the
    # file, as well as accessors to find current set and case.
    my $self = shift;
}

# Define set 'b', a call to 'test_set' is not necessary if you do not want
# to set any options.
sub set_b { ... }

# If you don't like the magic of finding test subs automatically
test_set c => sub { ... };

# the long and ugly version.
test_set d => (
    ...
    method => sub { ... },
);

test_case foo => (
    # Same as test_set() options except no test count.
    # Added:
        # Only run the specified sets
        only_sets => [qw//],
        # Do not run these sets
        skip_sets => [qw//],
);
sub case_foo {
    my $self = shift;
    # Setup tasks for this case...
}

# Cases have the same defenition styles and options as sets.

PROBLEMS ADDRESSED

forking code

This is partially solved by Test::More::Fork as well as Test::Fork and a few others. But no current solution is ideal.

Test::Suite will work with forking. When the module if first loaded it will record the current pid and create a listen socket. If a test is run under another pid it will connect to the root process's socket and send it the results as opposed to printing the TAP output itself.

perl tests are a collection of scripts

Currently perl tests are usually a collection of scripts under t/ that hopefully follow a useful or meaningful naming scheme. Scripts start, run their tests in order, and exit. Running specific groups of tests requires hacking on the script each time.

test randomisation

Cases and sets will be run in random order unless random is turned off when importing Test::Suite. Random can also be tunred off using a command line option. Every set will be run once per case.

running only specified tests as well as all

Must be able to run specific cases/sets

$ prove_suite TestModule [Case(s)] [Set(s)]
test grouping

Tests will be divided into sets. Sets can be run individually as well as in sequence. Each test will also have an initialization method to do initial setup. Test classes will also have cases, every set will be run once per case.

handle death

Each case and each set within each case will be run using eval or Try::Tiny so that a single test that blows up will not kill everything. If killing everything is desired then you can use on_fail => 'abort' in your case/set defenitions.

If a set or case dies (outside of an eval or dies_ok) it will be considered a single test failure.

tests mirror code

Test::Suite tests will be modules with a package name nearly identical to the package being tested.

Options (TIMTOWTDI)

Module:
lib/App/MyApp.pm

Options:
ts/App/MyApp.pm
lib/App/TEST/MyApp.pm

Tests can mirror lib inside /ts (/t should be reserved for script based tests and prove). They can also be in an uppercase 'TEST' directory within the directory containing the tested module.

You will need to configure Module::Build or Module::Install to ignore your test files for installation (unless you really don't care, but please do).

Module::Build and Module::Install do not directly support running Test::Suite tests, so you can create a test.pl test script that runs the tests, or a t/suite.t which does the same.

There should probably be extensions to the install/build modules to make Test::Suite work.

test reporting

When a test fails it should provide the filename, line number, case name and set name. As well in cases sich as is_deeply it should list all the differences at the level that breaks (though not all deep differences)

most commonly needed test functions

All the functionality of Test::Simple, Test::More, Test::Exception, and Test::Warn should be exported by default. It is unfortunate, but most functions will likely need to be re-implemented to work with the forking capabilities.

coverage

prove_suite should have a simple flag to turn on coverage testing

AUTHORS

Chad Granum exodist7@gmail.com

COPYRIGHT

Copyright (C) 2010 Chad Granum

Test-Suite is free software; Standard perl licence.

Test-Suite 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.