/[Frey]/trunk/lib/Frey/Test/Runner.pm
This is repository of my old source code which isn't updated any more. Go to git.rot13.org for current projects!
ViewVC logotype

Annotation of /trunk/lib/Frey/Test/Runner.pm

Parent Directory Parent Directory | Revision Log Revision Log


Revision 570 - (hide annotations)
Thu Nov 27 22:11:13 2008 UTC (15 years, 4 months ago) by dpavlin
File size: 3968 byte(s)
tweaks all over the place
1 dpavlin 484 package Frey::Test::Runner;
2     use Moose;
3    
4     extends 'Frey';
5     with 'Frey::Web';
6 dpavlin 486 with 'Frey::Storage';
7 dpavlin 484
8     use TAP::Harness;
9     use TAP::Formatter::HTML;
10     use Data::Dump qw/dump/;
11 dpavlin 549 use File::Slurp;
12 dpavlin 484
13 dpavlin 486 use Frey::SVK;
14 dpavlin 489 use Frey::PPI;
15 dpavlin 486
16 dpavlin 484 has tests => (
17     is => 'rw',
18     isa => 'ArrayRef[Str]',
19     required => 1,
20     lazy => 1, # FIXME ask users which tests to run
21 dpavlin 486 default => sub {
22     [ Frey::SVK->modified ]
23     },
24 dpavlin 506 documentation => 'run tests which are result of modifications or whole full tests',
25 dpavlin 484 );
26    
27 dpavlin 506 has test => (
28     is => 'rw',
29     isa => 'Str',
30     documentation => 'run only this single test',
31     );
32    
33 dpavlin 528 has test_because => (
34     documentation => 'returns classes responsable for each test run',
35 dpavlin 506 is => 'rw',
36     # isa => 'HashRef[Hashref[Int]',
37     required => 1,
38     lazy => 1,
39     default => sub {
40     my $self = shift;
41     my $depends;
42    
43     # collect real tests
44     map {
45     $depends->{$_}->{'test modified'}++ if m{\.t$};
46     } @{ $self->tests };
47    
48     # and tests which depend on modified classes supplied
49     map {
50     if ( m{(.+)\.pm$} ) {
51     my $class = $1;
52     $class =~ s{^lib/}{};
53     $class =~ s{/}{::}g;
54     warn "extract tests from $_ class $class";
55     $depends->{$_}->{$class}++ foreach Frey::PPI->new( class => $class )->has_tests;
56     }
57     } @{ $self->tests };
58    
59     return $depends;
60     },
61     );
62    
63 dpavlin 484 sub as_markup {
64     my ($self) = @_;
65    
66 dpavlin 549 my $path = 'var/test/';
67     my $running_pid = "$path/running.pid";
68 dpavlin 519
69 dpavlin 549 my $pid = read_file $running_pid if -e $running_pid;
70     if ( $pid ) {
71     if ( kill 0, $pid ) {
72     warn "ABORTING: $self started twice";
73     return 'abort';
74     } else {
75     warn "got $pid from $running_pid but no process alive, ignoring...";
76     }
77     }
78    
79     write_file( $running_pid, $$ );
80     warn "# started $self with pid $$ -> $running_pid";
81    
82 dpavlin 484 my $f = TAP::Formatter::HTML->new({
83 dpavlin 491 # silent => 1,
84 dpavlin 484
85     inline_css => 1,
86 dpavlin 491 inline_js => 0,
87 dpavlin 484 });
88     my $h = TAP::Harness->new({
89     merge => 1,
90     formatter => $f,
91     });
92    
93 dpavlin 506 my @tests;
94 dpavlin 491
95 dpavlin 506 @tests = ( $self->test ) if $self->test;
96 dpavlin 491
97 dpavlin 528 if ( my $depends = $self->test_because ) {
98 dpavlin 507 @tests = grep {
99 dpavlin 511 $_ ne '' &&
100 dpavlin 507 ! m{$0} # break recursion
101     } sort keys %{ $depends } unless @tests;
102     }
103 dpavlin 489
104 dpavlin 528 $self->add_status( { test => { depends => $self->test_because } } );
105 dpavlin 506
106     if ( ! @tests ) {
107 dpavlin 528 warn "can't find any tests ", dump( $self->tests ), " within depends ", dump( $self->test_because );
108 dpavlin 565 # warn "running all tests instead";
109     # @tests = glob('t/*.t');
110     @tests = glob('t/01*.t'); # XXX default tests
111 dpavlin 506 }
112    
113 dpavlin 519 $self->title( join(' ', @tests ) );
114    
115 dpavlin 484 warn "testing ",dump( @tests );
116     $h->runtests( @tests );
117    
118 dpavlin 491 $self->store( 'var/test/' . time() . '.yaml', $h );
119 dpavlin 486
120 dpavlin 484 my $html = ${ $f->html };
121     # warn $html;
122     warn "got ",length($html), " bytes";
123 dpavlin 494
124     while ( $html =~ s{(<style.+?/style>)}{}gs ) {
125 dpavlin 512 my $style = $1;
126     $style =~ s[((?:body|html)\s+{[^}]+})][/\* $1 \*/]sg; # remove some styles
127     $self->add_head( $style );
128 dpavlin 494 }
129    
130     $self->add_head(qq|
131     <style type="text/css">
132     /* CSS to show-hide full text results */
133     ul.test-out { display: none; }
134     td.results:hover ul.test-out { display: block; }
135     </style>
136     |);
137 dpavlin 512 $html =~ s{<div id="menu">.+?</div>}{}sg; # remove menu which doesn't work without JavaScript
138 dpavlin 494
139     $html =~ s{^.*<body>}{}s;
140     $html =~ s{</body>.*$}{}s;
141    
142 dpavlin 497 $html =~ s{(<a class="file") href="#"(.+?)>t/(.+?)</a>}{<a target="editor" href="/editor+t/$3.t+1" name="t/$3.t" $2>$3</a>}sg;
143 dpavlin 494
144 dpavlin 508 $html = $self->editor_links( $html );
145    
146 dpavlin 528 if ( my $depends = $self->test_because ) {
147 dpavlin 508 $html .= qq|Test dependencies:|
148     . qq|<ul><li>|
149 dpavlin 497 . join("</li>\n<li>",
150     map {
151 dpavlin 519 qq|<a href="?test=$_"><tt>$_</tt></a> &larr; |
152 dpavlin 497 .
153     join(' ',
154     map {
155 dpavlin 503 if ( m{\s} ) {
156     $_ # human comment with space
157     } else {
158     qq|<a target="introspect" href="/$_" title="introspect">$_</a>|
159     # qq|<a target="editor" href="/editor+$_+1" title="edit">$_</a>|
160     }
161 dpavlin 528 } keys %{ $depends->{$_} }
162 dpavlin 497 )
163     } @tests )
164     . qq|</li></ul>|
165 dpavlin 494 ;
166 dpavlin 570 } else {
167     warn "# test_because empty";
168 dpavlin 508 }
169 dpavlin 527
170     $self->add_icon( $1 ) if $html =~ m{class="(passed|failed)"};
171    
172 dpavlin 549 unlink $running_pid or die "can't remove $running_pid: $!";
173    
174 dpavlin 519 return $html;
175 dpavlin 484 }
176    
177     1;

  ViewVC Help
Powered by ViewVC 1.1.26