Devel::CoverX::Covered::Db - Covered database collection and reporting


Devel-CoverX-Covered documentation Contained in the Devel-CoverX-Covered distribution.

Index


Code Index:

NAME

Top

Devel::CoverX::Covered::Db - Covered database collection and reporting

DESCRIPTION

Top

Error handling model

Failures will result in a die.

PROPERTIES

Top

dir

The directory for the cover_db.

db

DBIx::Simple $db object. Created lazily under "dir".

report_file

Subref called for each $file processed, $report_file->($file).

rex_skip_calling_file

Regex matching test files to skip.

Default: matches prove and prove.bat

rex_skip_source_file

Array ref with Regexes matching source files to skip.

Default: []

METHODS

Top

connect_to_db() : DBIx::Simple $db

Connect to the covered db and return the new DBIx::Simple object.

If there is no db at all, create it first.

create_db($db_file) : DBIx::Simple $db

Create $db_file with the correct schema.

Return newly created DBIx::Simple $db object.

schema_version() : $version_string

Return a version number of the schema.

Update this manually whenever the schema changes.

in_transaction($subref) : $ret

Run $subref->() in a transaction and return the return value of $subref in scalar context.

If anything dies inside $subref, roll back and rethrow exception.

collect_runs() : 1

Collect coverage statistics for test runs in "dir".

get_run_dirs() : @dirs

Return list of directories for test runs under the "dir".

collect_run($cover_db) : 1 | 0

Collect coverage statistics for the test run Devel::Cover::DB $cover_db.

Don't collect coverage for eval (-e), nor for any test file matching rex_skip_calling_file.

Return 1 if the test run was collected, else 0.

calling_file_name($cover_db) : $calling_file_name | ""

Extract the $calling_file_name from $cover_db and return it.

Return "" if it's not a suitable calling file, or the callinging file is on the skip list. Warn with specifics.

is_calling_file_name_valid($file_name) : 1 | 0

Return 1 if $file_name is a valid calling file name, else 0.

It may be invalid because of unsuitability, or because it was skipped.

is_source_file_name_valid($file_name) : 1 | 0

Return 1 if $file_name is a valid source file name, else 0.

It may be invalid because of unsuitability, or because it was skipped.

reset_calling_file($calling_file_name) : 1

Clear out the stored data related to $calling_file_name.

report_metric_coverage(metric_type, calling_file, covered_file, covered_row, covered_sub_name, metric) : 1

Report the coverage metric defined by the parameters.

Make all file paths relative to "dir" if possible.

get_file_id($file_name) : $file_id

Return the db id of the table "file" row for $file_name. Create a new row if missing.

Memoized.

Return the new or existing $file_id.

get_metric_type_id($metric_type) : $metric_type_id

Return the db id of the table "metric_type" row for $metric_type. Create a new row if missing.

Memoized.

Return the new or existing $metric_type_id.

get_lookup_table_id($table_name, $id_column_name, $value_column_name, $lookup_table_value) : $lookup_table_id

Return the db id in $id_column_name of the $table_name row for $lookup_table_value.

Create a new row if missing.

Return the new or existing $lookup_table_id.

test_files_covering($source_file_name, [$sub]) : @test_file_names

Return list of test files that cover any line in $source_file_name. Or if $sub is passed, limit to test files covering that sub.

test_files() : @test_file_names

Return list of test files in the db.

source_files_covered_by($test_file_name) : @source_file_names

Return list of source files that are covered by $test_file_name.

covered_files() : @source_file_names

Return list of source files in the db.

covered_subs($source_file_name) : @array_refs_sub_count

Return list of array refs [ $sub_name, $coverate_count ] with the subs in $source_file_name, and the accumulated coverage count for any test files covering it.

relative_file($file) : Path::Class::File $relative_file

Return $file relative to cover_db/.. if possible, otherwise just return $file.


Devel-CoverX-Covered documentation Contained in the Devel-CoverX-Covered distribution.
use strict;
package Devel::CoverX::Covered::Db;
use Moose;

use Devel::Cover::DB;

use Data::Dumper;
use DBIx::Simple;
use DBD::SQLite;
use File::Path;
use File::chdir;
use Path::Class;
use Memoize;



has 'dir' => (
    is => 'ro',
    isa => 'Path::Class::Dir',
    default => sub { Path::Class::dir("./cover_db")->absolute },
);




has 'db' => (
    is => 'ro',
    isa => 'DBIx::Simple',
    lazy => 1,
    default => sub { $_[0]->connect_to_db },
);



has 'report_file' => (
    is => 'ro',
    isa => 'CodeRef',
    default => sub { sub { } },
);



has 'rex_skip_calling_file' => (
    is => 'rw',
    isa => 'RegexpRef',
    default => sub { qr/ \b prove ( [.]bat )? $/x },
);



has 'rex_skip_source_file' => (
    is => 'rw',
    isa => 'ArrayRef[RegexpRef]',
    default => sub { [] },
);



sub connect_to_db {
    my $self = shift;

    my $db_file = file(
        $self->dir->parent,
        "covered",
        "covered-" . $self->schema_version . ".db",
    );

    -r $db_file or return $self->create_db($db_file);

    my $db = DBIx::Simple->connect("dbi:SQLite:dbname=$db_file", "", "", { RaiseError => 1 });

    return $db;
}



sub create_db {
    my $self = shift;
    my ($db_file) = @_;

    #Refactor: extract method: ensure_dir_of_file
    my $db_dir = $db_file->dir;
    mkpath([ $db_dir ]);
    -e $db_dir or die("Could not create covered db dir ($db_dir)\n");

    my $db = DBIx::Simple->connect("dbi:SQLite:dbname=$db_file", "", "", { RaiseError => 1 });

    my @ddl_tables = (
        q{
                        CREATE TABLE file (
                                file_id INTEGER PRIMARY KEY,

                                name               VARCHAR NOT NULL
                        )
                },
        q{
                        CREATE UNIQUE INDEX file_name ON file (name)
                },

        q{
                        CREATE TABLE metric_type (
                                metric_type_id INTEGER PRIMARY KEY,

                                type               VARCHAR NOT NULL
                        )
                },
        q{
                        CREATE UNIQUE INDEX metric_type_type ON metric_type (type)
                },

        q{
                        CREATE TABLE covered_calling_metric (
                                covered_calling_metric_id INTEGER PRIMARY KEY,

                                calling_file_id    INTEGER NOT NULL REFERENCES file (file_id),
                                covered_file_id    INTEGER NOT NULL REFERENCES file (file_id),
                                covered_row        INTEGER NOT NULL,
                                covered_sub_name   VARCHAR NOT NULL,
                                metric_type_id     INTEGER NOT NULL REFERENCES metric_type (metric_type_id),
                                metric             INTEGER NOT NULL
                        )
                },
        q{
                        CREATE INDEX covered_calling_metric_covered_metric_row ON covered_calling_metric (
                                covered_file_id,
                                metric_type_id,
                                metric,
                                covered_row
                        )
                },
        q{
                        CREATE INDEX covered_calling_metric_calling ON covered_calling_metric (calling_file_id)
                },


        ###REMEMBER: update schema_version when changing the schema
    );

    for my $ddl (@ddl_tables) {
#print "DDL: $ddl\n";
        $db->query($ddl) or die("Could not run DDL ($ddl): " . $db->error . "\n");
    }

    return $db;
}



sub schema_version {
    my $self = shift;
    "0.01";
}



sub in_transaction {
    my $self = shift;
    my ($subref) = @_;

    my $db = $self->db;

    $db->begin();
    my $ret = eval { $subref->() };
    if(my $err = $@) {
        $db->rollback();
        die $err;
    }
    else {
        $db->commit();
    }

    return $ret;
}



sub collect_runs {
    my $self = shift;

    $self->in_transaction( sub {
        local $CWD = $self->dir->parent;
        for my $run_db_dir ($self->get_run_dirs()) {
            my $cover_db = Devel::Cover::DB->new(db => $self->dir);

            my $run_db_file = "$run_db_dir/cover.12";  #Eeh, refactor
            -e $run_db_file or warn("No run db ($run_db_file)\n"), next;
            $cover_db->read($run_db_file);

            $self->collect_run($cover_db);

            #Force release of data structure with too high ref count
            delete $cover_db->{cover};
        }
    });

    return 1;
}




sub get_run_dirs {
    my $self = shift;
    return grep { -d $_ } grep { /\d$/ } sort glob($self->dir . "/runs/*");
}



sub collect_run {
    my $self = shift;
    my ($cover_db) = @_;

    my $calling_file_name = $self->calling_file_name($cover_db) or return 0;
    $self->report_file->($calling_file_name);

    $self->reset_calling_file($calling_file_name);

    my @source_file_names = $cover_db->cover->items;
    for my $source_file_name (@source_file_names) {
        $self->is_source_file_name_valid($source_file_name) or next;

        my $file_data = $cover_db->cover->file($source_file_name);

        for my $metric_type ("subroutine" ) { #time, branch, statement
            my $row_metric = $file_data->$metric_type or next;

            for my $row (keys %$row_metric) {
                my $row_locations = $row_metric->{$row} or next;
                my $row_location = $row_locations->[0] or next;

                my $sub_name = "";
                if($row_location->can("name")) {
                    $sub_name = $row_location->name;
                    $sub_name eq "BEGIN" and next;
                    $sub_name eq "__ANON__" and next;
                }

                my $is_covered = $row_location->covered;

                $self->report_metric_coverage(
                    metric_type      => $metric_type,
                    calling_file     => $calling_file_name,
                    covered_file     => $source_file_name,
                    covered_row      => $row,
                    covered_sub_name => $sub_name,
                    metric           => $is_covered,
                );
            }
        }
    }

    return 1;
}




sub calling_file_name {
    my $self = shift;
    my ($cover_db) = @_;

    my @runs = $cover_db->runs;
    @runs > 1 and warn("More than one run in run cover db\n"), return "";
    my $calling_file_name = $runs[0]->run;

    $self->is_calling_file_name_valid($calling_file_name)
            or warn("Skipping test file ($calling_file_name)\n"), return "";

    return $calling_file_name;
}



sub is_calling_file_name_valid {
    my $self = shift;
    my ($calling_file_name) = @_;

    $calling_file_name eq "-e" and return 0;
    $calling_file_name =~ $self->rex_skip_calling_file and return 0;

    return 1;
}



sub is_source_file_name_valid {
    my $self = shift;
    my ($source_file_name) = @_;

    for my $rex (@{$self->rex_skip_source_file}) {
        $source_file_name =~ $rex and return 0;
    }

    return 1;
}



sub reset_calling_file {
    my $self = shift;
    my ($calling_file_name) = @_;

    my $calling_file_id = $self->get_file_id($calling_file_name);
    $self->db->delete("covered_calling_metric", { calling_file_id => $calling_file_id });

    return 1;
}



sub report_metric_coverage {
    my $self = shift;
    my (%p) = @_;

    $p{metric} ||= 0;
    $p{$_} = $self->relative_file($p{$_}) . "" for (qw/ calling_file covered_file /);

    my $metric_lookup_method = {
        calling_file => "get_file_id",
        covered_file => "get_file_id",
        metric_type  => "get_metric_type_id",
    };
    for my $metric ( keys %$metric_lookup_method ) {
        my $lookup_method = $metric_lookup_method->{$metric};
        $p{ "${metric}_id" } = $self->$lookup_method( $p{$metric} );
        delete $p{$metric};
    }

    $self->db->insert("covered_calling_metric", \%p);

    return 1;
}



memoize("get_file_id");
sub get_file_id {
    my $self = shift;
    my ($file_name) = @_;
    $self->get_lookup_table_id("file", "file_id", "name", $file_name);
}



memoize("get_metric_type_id");
sub get_metric_type_id {
    my $self = shift;
    my ($metric_type) = @_;
    $self->get_lookup_table_id("metric_type", "metric_type_id", "type", $metric_type);
}



sub get_lookup_table_id {
    my $self = shift;
    my ($table_name, $id_column_name, $value_column_name, $lookup_table_value) = @_;

    $self->db->query(
        "select $id_column_name from $table_name where $value_column_name = ?",
        $lookup_table_value,
    )->into( my $lookup_table_id );
    $lookup_table_id and return $lookup_table_id;

    $self->db->insert($table_name, { $value_column_name => $lookup_table_value });
    $lookup_table_id = $self->db->last_insert_id(undef, undef, $table_name, "$id_column_name");

    return $lookup_table_id;
}



sub test_files_covering {
    my $self = shift;
    my ($calling_file_name, $sub) = @_;

    #This needs to be sub coverage, so a "use" doesn't execute
    #statements e.g. "use strict;".

    my ($and_sub_name, @and_sub_name) = ("", ());
    if($sub) {
        ($and_sub_name, @and_sub_name) =  ("and ccm.covered_sub_name = ?", ($sub));
    }

    my @test_files = $self->db->query(
        qq{
                SELECT DISTINCT(f_calling.name)
                        FROM covered_calling_metric ccm, file f_calling, file f_covered
                        WHERE
                                        f_covered.name = ?
                                AND ccm.calling_file_id = f_calling.file_id
                                AND ccm.covered_file_id = f_covered.file_id
                                AND ccm.metric_type_id = ?
                                AND ccm.metric > 0
                                $and_sub_name
                        ORDER by f_calling.name
                },
        $calling_file_name,
        $self->get_metric_type_id("subroutine"),
        @and_sub_name,
    )->flat;

    return @test_files;
}



sub test_files {
    my $self = shift;

    my @test_files = $self->db->query(
        q{
                SELECT DISTINCT(f.name)
                        FROM covered_calling_metric ccm, file f
                        WHERE ccm.calling_file_id = f.file_id
                        ORDER by f.name
                },
    )->flat;

    return @test_files;
}



sub source_files_covered_by {
    my $self = shift;
    my ($test_file_name) = @_;

    #This needs to be sub coverage, so a "use" doesn't execute
    #statements e.g. "use strict;".
    my @source_files = $self->db->query(
        q{
                SELECT DISTINCT(f_covered.name)
                        FROM covered_calling_metric ccm, file f_calling, file f_covered
                        WHERE
                                        f_calling.name = ?
                                AND ccm.covered_file_id = f_covered.file_id
                                AND ccm.calling_file_id = f_calling.file_id
                                AND ccm.metric_type_id = ?
                                AND ccm.metric > 0
                        ORDER by f_covered.name
                },
        $test_file_name,
        $self->get_metric_type_id("subroutine"),
    )->flat;

    return @source_files;
}



sub covered_files {
    my $self = shift;

    my @source_files = $self->db->query(
        q{
                SELECT DISTINCT(f.name)
                        FROM covered_calling_metric ccm, file f
                        WHERE ccm.covered_file_id = f.file_id
                        ORDER by f.name
                },
    )->flat;

    return @source_files;
}



sub covered_subs {
    my $self = shift;
    my ($covered_file_name) = @_;

    my @sub_count = map { [ $_->[1], $_->[2] ] } $self->db->query(
        q{
                SELECT ccm.covered_row, ccm.covered_sub_name, SUM(ccm.metric)
                        FROM covered_calling_metric ccm, file f_covered
                        WHERE
                                        f_covered.name = ?
                                AND ccm.covered_file_id = f_covered.file_id
                                AND ccm.metric_type_id = ?
                        GROUP BY ccm.covered_row, ccm.covered_sub_name
                        ORDER by ccm.covered_row, ccm.covered_sub_name
                },
        $covered_file_name,
        $self->get_metric_type_id("subroutine"),
    )->arrays;

    return @sub_count;
}



sub relative_file {
    my $self = shift;
    my ($file) = @_;
    $file = file($file);
    $file->is_absolute or return $file;
    return $file->relative( $self->dir->parent );
}



1;



__END__