File Coverage

blib/lib/Devel/KYTProf.pm
Criterion Covered Total %
statement 119 138 86.2
branch 30 54 55.5
condition 6 33 18.1
subroutine 17 18 94.4
pod 0 5 0.0
total 172 248 69.3


line stmt bran cond sub pod time code
1             package Devel::KYTProf;
2 6     6   406359 use strict;
  6         57  
  6         176  
3 6     6   32 use warnings;
  6         10  
  6         433  
4              
5             our $VERSION = '0.9992';
6              
7             my $Applied = {};
8              
9             use Class::Data::Lite (
10 6         92 rw => {
11             namespace_regex => undef,
12             ignore_class_regex => undef,
13             context_classes_regex => undef,
14             logger => undef,
15             threshold => undef,
16             remove_linefeed => undef,
17             remove_escape_sequences => undef,
18              
19             color_time => 'red',
20             color_module => 'cyan',
21             color_info => 'blue',
22             color_call => 'green',
23              
24             _orig_code => {},
25             _prof_code => {},
26             },
27 6     6   2899 );
  6         3489  
28              
29 6     6   4207 use Module::Load ();
  6         6764  
  6         134  
30 6     6   3005 use Time::HiRes;
  6         8051  
  6         25  
31 6     6   4657 use Term::ANSIColor;
  6         50592  
  6         7056  
32              
33             sub import {
34 7     7   74 __PACKAGE__->apply_prof('DBI');
35 7         35 __PACKAGE__->apply_prof('LWP::UserAgent');
36 7         31 __PACKAGE__->apply_prof('Cache::Memcached::Fast');
37 7         31 __PACKAGE__->apply_prof('MogileFS::Client');
38 7         29 __PACKAGE__->apply_prof('Furl::HTTP');
39 7         3749 1;
40             }
41              
42             sub apply_prof {
43 35     35 0 95 my ($class, $pkg, $prof_pkg, @args) = @_;
44 35         52 eval { Module::Load::load($pkg) };
  35         86  
45 35 100       123557 return if $@;
46              
47 7   33     79 $prof_pkg ||= "Devel::KYTProf::Profiler::$pkg";
48 7         15 eval {Module::Load::load($prof_pkg)};
  7         32  
49 7 50       138 if ($@) {
50 0         0 die qq{failed to load profiler package "$prof_pkg" for "$pkg": $@\n};
51             }
52 7 50       118 unless ($prof_pkg->can('apply')) {
53 0         0 die qq{"$prof_pkg" has no `apply` method. A profiler package should implement it.\n};
54             }
55 7 100       45 return if ++$Applied->{$prof_pkg} > 1; # skip if already applied
56 6         30 $prof_pkg->apply(@args);
57             }
58              
59             sub add_profs {
60 7     7 0 149 my ($class, $module, $methods, $callback, $sampler) = @_;
61 7         14 eval {Module::Load::load($module)};
  7         34  
62 7 50       2038 if ($methods eq ':all') {
63 0         0 eval { Module::Load::load('Class/Inspector.pm') };
  0         0  
64 0 0       0 return if $@;
65 0         0 $methods = [];
66 0         0 @$methods = @{Class::Inspector->methods($module, 'public')};
  0         0  
67             }
68 7         24 for my $method (@$methods) {
69 26         84 $class->add_prof($module, $method, $callback, $sampler);
70             }
71             }
72              
73             sub add_prof {
74 40     40 0 317 my ($class, $module, $method, $callback, $sampler) = @_;
75 40         65 eval {Module::Load::load($module)};
  40         111  
76 40         9997 my $orig = $class->_orig_code->{$module}{$method};
77 40 50       301 unless ($orig) {
78 40 50       415 $orig = $module->can($method) or return;
79 40         161 $class->_orig_code->{$module}->{$method} = $orig;
80             }
81              
82             my $code = sub {
83 12 100   12   6113 if ($sampler) {
84 3         10 my $is_sample = $sampler->($orig, @_);
85 3 50       8 unless ($is_sample) {
86 0         0 return $orig->(@_);
87             }
88             }
89              
90 12         25 my ($package, $file, $line, $level);
91 12         58 my $namespace_regex = $class->namespace_regex;
92 12         80 my $ignore_class_regex = $class->ignore_class_regex;
93 12         68 my $context_classes_regex = $class->context_classes_regex;
94 12         63 my $threshold = $class->threshold;
95 12 50 33     96 if ($namespace_regex || $context_classes_regex) {
96 0         0 for my $i (1..30) {
97 0 0       0 my ($p, $f, $l) = caller($i) or next;
98 0 0 0     0 if (
      0        
      0        
      0        
99             $namespace_regex
100             &&
101             !$package
102             &&
103             $p =~ /^($namespace_regex)/
104             &&
105             (! $ignore_class_regex || $p !~ /$ignore_class_regex/)
106             ) {
107 0         0 ($package, $file, $line) = ($p, $f, $l);
108             }
109              
110 0 0 0     0 if ($context_classes_regex && !$level && $p =~ /^($context_classes_regex)$/) {
      0        
111 0         0 $level = $i;
112             }
113             }
114             } else {
115 12         42 for my $i (1..30) {
116 360 50       719 my ($p, $f, $l) = caller($i) or next;
117 0 0       0 if ($p !~ /^($module)/) {
118 0         0 ($package, $file, $line) = ($p, $f, $l);
119 0         0 last;
120             }
121             }
122             }
123 12 50       30 unless ($package) {
124 12         42 ($package, $file, $line) = caller;
125             }
126 12         82 my $start = [ Time::HiRes::gettimeofday ];
127 12         22 my ($res, @res);
128 12 50       31 if (wantarray) {
129 0         0 @res = $orig->(@_);
130             } else {
131 12         42 $res = $orig->(@_);
132             }
133 12         2150 my $ns = Time::HiRes::tv_interval($start) * 1000;
134 12 50 33     227 if (!$threshold || $ns >= $threshold) {
135 12         24 my $message = "";
136 12         157 $message .= colored(sprintf('% 9.3f ms ', $ns), $class->color_time);
137 12   50     344 $message .= colored(sprintf(' [%s] ', ref $_[0] || $_[0] || ''), $class->color_module);
138 12         194 my $cb_info;
139             my $cb_data;
140 12 100       29 if ($callback) {
141 7         21 my $v = $callback->($orig, @_);
142 7 100       42 if (ref $v eq "ARRAY") {
143 6         13 $cb_info = sprintf $v->[0], map { $v->[2]->{$_} } @{$v->[1]};
  12         40  
  6         16  
144 6         27 $cb_data = $v->[2];
145             } else {
146 1         3 $cb_info = $v;
147 1         3 $cb_data = {};
148             }
149             } else {
150 5         9 $cb_info = $method;
151 5         8 $cb_data = {};
152             }
153 12 100       36 $cb_info =~ s/[[:cntrl:]]//smg if $class->remove_escape_sequences;
154 12         100 $message .= colored(sprintf(' %s ', $cb_info), $class->color_info);
155 12         254 $message .= ' | ';
156 12   50     71 $message .= colored(sprintf('%s:%d', $package || '', $line || 0), $class->color_call);
      50        
157 12 50       193 $message =~ s/\n/ /g if $class->remove_linefeed;
158 12         68 $message .= "\n";
159 12 50       29 $class->logger ? $class->logger->log(
160             level => 'debug',
161             message => $message,
162             module => $module,
163             method => $method,
164             time => $ns,
165             package => $package,
166             file => $file,
167             line => $line,
168             data => $cb_data,
169             ) : print STDERR $message;
170             }
171 12 50       154 return wantarray ? @res : $res;
172 40         529 };
173 40         144 $class->_prof_code->{$module}->{$method} = $code;
174              
175 40         260 $class->_inject_code($module, $method, $code);
176             }
177              
178             sub _inject_code {
179 46     46   165 my ($class, $module, $method, $code) = @_;
180 6     6   243 no strict 'refs';
  6         18  
  6         314  
181 6     6   38 no warnings qw/redefine prototype/;
  6         11  
  6         1805  
182 46         69 *{"$module\::$method"} = $code;
  46         283  
183             }
184              
185             sub mute {
186 2     2 0 1491 my ($class, $module, @methods) = @_;
187              
188 2 100       8 if (scalar(@methods)) {
189 1         3 for my $method (@methods) {
190 1         4 $class->_inject_code($module, $method, $class->_orig_code->{$module}->{$method});
191             }
192             } else {
193 1         2 for my $method (keys %{$class->_orig_code->{$module}}) {
  1         4  
194 2         15 $class->_inject_code($module, $method, $class->_orig_code->{$module}->{$method});
195             }
196             }
197             }
198              
199             sub unmute {
200 2     2 0 1110 my ($class, $module, @methods) = @_;
201              
202 2 100       7 if (scalar(@methods)) {
203 1         3 for my $method (@methods) {
204 1         5 $class->_inject_code($module, $method, $class->_prof_code->{$module}->{$method});
205             }
206             } else {
207 1         2 for my $method (keys %{$class->_prof_code->{$module}}) {
  1         4  
208 2         13 $class->_inject_code($module, $method, $class->_prof_code->{$module}->{$method});
209             }
210             }
211             }
212              
213             {
214 6     6   53 no warnings 'redefine';
  6         12  
  6         433  
215       0     *DB::DB = sub {};
216             }
217              
218             1;
219              
220             __END__