File Coverage

blib/lib/Log/ger/Util.pm
Criterion Covered Total %
statement 100 170 58.8
branch 36 84 42.8
condition 3 14 21.4
subroutine 17 29 58.6
pod 0 18 0.0
total 156 315 49.5


line stmt bran cond sub pod time code
1             package Log::ger::Util;
2              
3             our $AUTHORITY = 'cpan:PERLANCAR'; # AUTHORITY
4             our $DATE = '2020-03-11'; # DATE
5             our $DIST = 'Log-ger'; # DIST
6             our $VERSION = '0.037'; # VERSION
7              
8 8     8   247615 use strict;
  8         45  
  8         198  
9 8     8   34 use warnings;
  8         12  
  8         390  
10              
11             require Log::ger;
12             require Log::ger::Heavy;
13              
14             sub _dump {
15 0 0   0   0 unless ($Log::ger::_dumper) {
16 0         0 eval {
17 8     8   41 no warnings 'once';
  8         11  
  8         489  
18 0         0 require Data::Dmp;
19 0         0 $Data::Dmp::OPT_REMOVE_PRAGMAS = 1;
20 0         0 1;
21             };
22 0 0       0 if ($@) {
23 8     8   54 no warnings 'once';
  8         17  
  8         3115  
24 0         0 require Data::Dumper;
25             $Log::ger::_dumper = sub {
26 0     0   0 local $Data::Dumper::Terse = 1;
27 0         0 local $Data::Dumper::Indent = 0;
28 0         0 local $Data::Dumper::Useqq = 1;
29 0         0 local $Data::Dumper::Deparse = 1;
30 0         0 local $Data::Dumper::Quotekeys = 0;
31 0         0 local $Data::Dumper::Sortkeys = 1;
32 0         0 local $Data::Dumper::Trailingcomma = 1;
33 0         0 local $Data::Dumper::Useqq = 1; # to show "\034", possible bug in Data::Dumper
34 0         0 Data::Dumper::Dumper($_[0]);
35 0         0 };
36             } else {
37 0     0   0 $Log::ger::_dumper = sub { Data::Dmp::dmp($_[0]) };
  0         0  
38             }
39             }
40 0         0 $Log::ger::_dumper->($_[0]);
41             }
42              
43             sub numeric_level {
44 15     15 0 926 my $level = shift;
45 15 100       86 return $level if $level =~ /\A\d+\z/;
46             return $Log::ger::Levels{$level}
47 2 50       10 if defined $Log::ger::Levels{$level};
48             return $Log::ger::Level_Aliases{$level}
49 0 0       0 if defined $Log::ger::Level_Aliases{$level};
50 0         0 die "Unknown level '$level'";
51             }
52              
53             sub string_level {
54 3     3 0 2131 my $level = shift;
55 3 100       11 return $level if defined $Log::ger::Levels{$level};
56             $level = $Log::ger::Level_Aliases{$level}
57 2 100       6 if defined $Log::ger::Level_Aliases{$level};
58 2         8 for (keys %Log::ger::Levels) {
59 9         12 my $v = $Log::ger::Levels{$_};
60 9 100       18 return $_ if $v == $level;
61             }
62 0         0 die "Unknown level '$level'";
63             }
64              
65             sub set_level {
66 8     8   50 no warnings 'once';
  8         14  
  8         550  
67 11     11 0 16105 $Log::ger::Current_Level = numeric_level(shift);
68 11         25 reinit_all_targets();
69             }
70              
71             sub _action_on_hooks {
72 8     8   46 no warnings 'once';
  8         21  
  8         8508  
73              
74 23     23   61 my ($action, $target_type, $target_name, $phase) = splice @_, 0, 4;
75              
76 23 50       58 my $hooks = $Log::ger::Global_Hooks{$phase} or die "Unknown phase '$phase'";
77 23 100       84 if ($target_type eq 'package') {
    50          
    50          
78 1   50     6 $hooks = ($Log::ger::Per_Package_Hooks{$target_name}{$phase} ||= []);
79             } elsif ($target_type eq 'object') {
80 0         0 my ($addr) = $target_name =~ $Log::ger::re_addr;
81 0   0     0 $hooks = ($Log::ger::Per_Object_Hooks{$addr}{$phase} ||= []);
82             } elsif ($target_type eq 'hash') {
83 0         0 my ($addr) = $target_name =~ $Log::ger::re_addr;
84 0   0     0 $hooks = ($Log::ger::Per_Hash_Hooks{$addr}{$phase} ||= []);
85             }
86              
87 23 100       65 if ($action eq 'add') {
    50          
    50          
    0          
    0          
    0          
88 20         34 my $hook = shift;
89             # XXX remove duplicate key
90             # my $key = $hook->[0];
91 20         54 unshift @$hooks, $hook;
92             } elsif ($action eq 'remove') {
93 0         0 my $code = shift;
94 0         0 for my $i (reverse 0..$#{$hooks}) {
  0         0  
95 0 0       0 splice @$hooks, $i, 1 if $code->($hooks->[$i]);
96             }
97             } elsif ($action eq 'reset') {
98 3         6 my $saved = [@$hooks];
99             splice @$hooks, 0, scalar(@$hooks),
100 3         5 @{ $Log::ger::Default_Hooks{$phase} };
  3         10  
101 3         9 return $saved;
102             } elsif ($action eq 'empty') {
103 0         0 my $saved = [@$hooks];
104 0         0 splice @$hooks, 0;
105 0         0 return $saved;
106             } elsif ($action eq 'save') {
107 0         0 return [@$hooks];
108             } elsif ($action eq 'restore') {
109 0         0 my $saved = shift;
110 0         0 splice @$hooks, 0, scalar(@$hooks), @$saved;
111 0         0 return $saved;
112             }
113             }
114              
115             sub add_hook {
116 19     19 0 35 my ($phase, $hook) = @_;
117 19         39 _action_on_hooks('add', '', undef, $phase, $hook);
118             }
119              
120             sub add_per_target_hook {
121 1     1 0 2 my ($target_type, $target_name, $phase, $hook) = @_;
122 1         2 _action_on_hooks('add', $target_type, $target_name, $phase, $hook);
123             }
124              
125             sub remove_hook {
126 0     0 0 0 my ($phase, $code) = @_;
127 0         0 _action_on_hooks('remove', '', undef, $phase, $code);
128             }
129              
130             sub remove_per_target_hook {
131 0     0 0 0 my ($target_type, $target_name, $phase, $code) = @_;
132 0         0 _action_on_hooks('remove', $target_type, $target_name, $phase, $code);
133             }
134              
135             sub reset_hooks {
136 3     3 0 4258 my ($phase) = @_;
137 3         10 _action_on_hooks('reset', '', undef, $phase);
138             }
139              
140             sub reset_per_target_hooks {
141 0     0 0 0 my ($target_type, $target_name, $phase) = @_;
142 0         0 _action_on_hooks('reset', $target_type, $target_name, $phase);
143             }
144              
145             sub empty_hooks {
146 0     0 0 0 my ($phase) = @_;
147 0         0 _action_on_hooks('empty', '', undef, $phase);
148             }
149              
150             sub empty_per_target_hooks {
151 0     0 0 0 my ($target_type, $target_name, $phase) = @_;
152 0         0 _action_on_hooks('empty', $target_type, $target_name, $phase);
153             }
154              
155             sub save_hooks {
156 0     0 0 0 my ($phase) = @_;
157 0         0 _action_on_hooks('save', '', undef, $phase);
158             }
159              
160             sub save_per_target_hooks {
161 0     0 0 0 my ($target_type, $target_name, $phase) = @_;
162 0         0 _action_on_hooks('save', $target_type, $target_name, $phase);
163             }
164              
165             sub restore_hooks {
166 0     0 0 0 my ($phase, $saved) = @_;
167 0         0 _action_on_hooks('restore', '', undef, $phase, $saved);
168             }
169              
170             sub restore_per_target_hooks {
171 0     0 0 0 my ($target_type, $target_name, $phase, $saved) = @_;
172 0         0 _action_on_hooks('restore', $target_type, $target_name, $phase, $saved);
173             }
174              
175             sub reinit_target {
176 1     1 0 2 my ($target_type, $target_name) = @_;
177              
178             # adds target if not already exists
179 1         3 Log::ger::add_target($target_type, $target_name, {}, 0);
180              
181 1 50       2 if ($target_type eq 'package') {
    0          
    0          
182 1         1 my $per_target_conf = $Log::ger::Package_Targets{$target_name};
183 1         3 Log::ger::init_target(package => $target_name, $per_target_conf);
184             } elsif ($target_type eq 'object') {
185 0 0       0 my ($obj_addr) = $target_name =~ $Log::ger::re_addr
186             or die "Invalid object '$target_name': not a reference";
187 0 0       0 my $v = $Log::ger::Object_Targets{$obj_addr}
188             or die "Unknown object target '$target_name'";
189 0         0 Log::ger::init_target(object => $v->[0], $v->[1]);
190             } elsif ($target_type eq 'hash') {
191 0 0       0 my ($hash_addr) = $target_name =~ $Log::ger::re_addr
192             or die "Invalid hashref '$target_name': not a reference";
193 0 0       0 my $v = $Log::ger::Hash_Targets{$hash_addr}
194             or die "Unknown hash target '$target_name'";
195 0         0 Log::ger::init_target(hash => $v->[0], $v->[1]);
196             } else {
197 0         0 die "Unknown target type '$target_type'";
198             }
199             }
200              
201             sub reinit_all_targets {
202 28     28 0 60 for my $pkg (keys %Log::ger::Package_Targets) {
203             #print "D:reinit package $pkg\n";
204             Log::ger::init_target(
205 28         87 package => $pkg, $Log::ger::Package_Targets{$pkg});
206             }
207 28         72 for my $k (keys %Log::ger::Object_Targets) {
208 18         19 my ($obj, $per_target_conf) = @{ $Log::ger::Object_Targets{$k} };
  18         36  
209 18         29 Log::ger::init_target(object => $obj, $per_target_conf);
210             }
211 28         198 for my $k (keys %Log::ger::Hash_Targets) {
212 4         5 my ($hash, $per_target_conf) = @{ $Log::ger::Hash_Targets{$k} };
  4         9  
213 4         8 Log::ger::init_target(hash => $hash, $per_target_conf);
214             }
215             }
216              
217             sub set_plugin {
218 18     18 0 53 my %args = @_;
219              
220 18         27 my $hooks;
221 18 50       41 if ($args{hooks}) {
222 0         0 $hooks = $args{hooks};
223             } else {
224 8     8   58 no strict 'refs';
  8         12  
  8         3776  
225 18   50     47 my $prefix = $args{prefix} || 'Log::ger::Plugin::';
226 18         31 my $mod = $args{name};
227 18 50       63 $mod = $prefix . $mod unless index($mod, $prefix) == 0;
228 18         98 (my $mod_pm = "$mod.pm") =~ s!::!/!g;
229 18         4580 require $mod_pm;
230 18 50       200 my $meta = $mod->can("meta") ? $mod->meta : {v=>1};
231 18   50     60 my $v = $meta->{v} || 1;
232              
233             # history of v bumping:
234             #
235             # - v increased from 1 to 2 in Log::ger v0.037 to force all plugins that
236             # were not compatible with Log::ger 0.032 (removed
237             # create_logml_routine phase) to be upgraded.
238              
239 18 50       40 unless ($v == 2) {
240             die "Plugin '$mod' (version ".(${"$mod\::VERSION"} || "dev").")".
241             " follows meta version $v but Log::ger (version ".
242 0   0     0 (${__PACKAGE__."::VERSION"} || "dev").
      0        
243             ") (>0.032) requires meta version 2, ".
244             "please upgrade the plugin first";
245             }
246 18 50       26 $hooks = &{"$mod\::get_hooks"}(%{ $args{conf} || {} });
  18         68  
  18         60  
247             }
248              
249             {
250 18 100       30 last unless $args{replace_package_regex};
  18         45  
251 16         21 my $all_hooks;
252 16 50       36 if (!$args{target}) {
    0          
    0          
    0          
253 16         26 $all_hooks = \%Log::ger::Global_Hooks;
254             } elsif ($args{target} eq 'package') {
255 0         0 $all_hooks = $Log::ger::Per_Package_Hooks{ $args{target_arg} };
256             } elsif ($args{target} eq 'object') {
257 0         0 my ($addr) = $args{target_arg} =~ $Log::ger::re_addr;
258 0         0 $all_hooks = $Log::ger::Per_Object_Hooks{$addr};
259             } elsif ($args{target} eq 'hash') {
260 0         0 my ($addr) = $args{target_arg} =~ $Log::ger::re_addr;
261 0         0 $all_hooks = $Log::ger::Per_Hash_Hooks{$addr};
262             }
263 16 50       36 last unless $all_hooks;
264 16         59 for my $phase (keys %$all_hooks) {
265 160         205 my $hooks = $all_hooks->{$phase};
266 160         160 for my $i (reverse 0..$#{$hooks}) {
  160         234  
267             splice @$hooks, $i, 1
268 71 100       255 if $hooks->[$i][0] =~ $args{replace_package_regex};
269             }
270             }
271             }
272              
273 18         46 for my $phase (keys %$hooks) {
274 20         34 my $hook = $hooks->{$phase};
275 20 100       41 if (defined $args{target}) {
276             add_per_target_hook(
277 1         3 $args{target}, $args{target_arg}, $phase, $hook);
278             } else {
279 19         43 add_hook($phase, $hook);
280             }
281             }
282              
283 18         35 my $reinit = $args{reinit};
284 18 50       47 $reinit = 1 unless defined $reinit;
285 18 50       40 if ($reinit) {
286 18 100       32 if (defined $args{target}) {
287 1         2 reinit_target($args{target}, $args{target_arg});
288             } else {
289 17         40 reinit_all_targets();
290             }
291             }
292             }
293              
294             1;
295             # ABSTRACT: Utility routines for Log::ger
296              
297             __END__