line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package UR::Namespace::Command::Show::Subclasses; |
2
|
|
|
|
|
|
|
|
3
|
1
|
|
|
1
|
|
23
|
use strict; |
|
1
|
|
|
|
|
1
|
|
|
1
|
|
|
|
|
28
|
|
4
|
1
|
|
|
1
|
|
3
|
use warnings; |
|
1
|
|
|
|
|
1
|
|
|
1
|
|
|
|
|
23
|
|
5
|
1
|
|
|
1
|
|
4
|
use UR; |
|
1
|
|
|
|
|
1
|
|
|
1
|
|
|
|
|
5
|
|
6
|
1
|
|
|
1
|
|
382
|
use YAML; |
|
1
|
|
|
|
|
2980
|
|
|
1
|
|
|
|
|
1027
|
|
7
|
|
|
|
|
|
|
|
8
|
|
|
|
|
|
|
my $spacing = ''; |
9
|
|
|
|
|
|
|
|
10
|
|
|
|
|
|
|
class UR::Namespace::Command::Show::Subclasses { |
11
|
|
|
|
|
|
|
is => 'Command::V2', |
12
|
|
|
|
|
|
|
has=> [ |
13
|
|
|
|
|
|
|
superclass => { |
14
|
|
|
|
|
|
|
is => 'Text', |
15
|
|
|
|
|
|
|
shell_args_position => 1, |
16
|
|
|
|
|
|
|
doc => 'Only show subclasses of this class.', |
17
|
|
|
|
|
|
|
}, |
18
|
|
|
|
|
|
|
color => { |
19
|
|
|
|
|
|
|
is => 'Boolean', |
20
|
|
|
|
|
|
|
is_optional => 1, |
21
|
|
|
|
|
|
|
default_value => 1, |
22
|
|
|
|
|
|
|
doc => 'Display in color.', |
23
|
|
|
|
|
|
|
}, |
24
|
|
|
|
|
|
|
maximum_depth => { |
25
|
|
|
|
|
|
|
is => 'Int', |
26
|
|
|
|
|
|
|
is_optional => 1, |
27
|
|
|
|
|
|
|
default_value => -1, |
28
|
|
|
|
|
|
|
doc => 'Maximum subclass depth. Negative means infinite.', |
29
|
|
|
|
|
|
|
}, |
30
|
|
|
|
|
|
|
recalculate => { |
31
|
|
|
|
|
|
|
is => 'Boolean', |
32
|
|
|
|
|
|
|
is_optional => 1, |
33
|
|
|
|
|
|
|
default_value => 0, |
34
|
|
|
|
|
|
|
doc => 'Recreate the cache instead of using the results of a previous run.', |
35
|
|
|
|
|
|
|
}, |
36
|
|
|
|
|
|
|
flat => { |
37
|
|
|
|
|
|
|
is => 'Boolean', |
38
|
|
|
|
|
|
|
is_optional => 1, |
39
|
|
|
|
|
|
|
doc => 'Simply prints the subclass names with no other formatting or coloring.', |
40
|
|
|
|
|
|
|
} |
41
|
|
|
|
|
|
|
], |
42
|
|
|
|
|
|
|
doc => 'Display subclasses of a given class.', |
43
|
|
|
|
|
|
|
}; |
44
|
|
|
|
|
|
|
|
45
|
|
|
|
|
|
|
sub help_synopsis { |
46
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
47
|
0
|
|
|
|
|
|
my $result .= <
|
48
|
|
|
|
|
|
|
Displays a tree of subclasses of a given class. |
49
|
|
|
|
|
|
|
EOP |
50
|
0
|
|
|
|
|
|
return $result; |
51
|
|
|
|
|
|
|
} |
52
|
|
|
|
|
|
|
|
53
|
|
|
|
|
|
|
sub help_detail { |
54
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
55
|
0
|
|
|
|
|
|
my $result .= <
|
56
|
|
|
|
|
|
|
Displays a tree containing the names (and optionally other info) of the subclasses |
57
|
|
|
|
|
|
|
of a given class. |
58
|
|
|
|
|
|
|
|
59
|
|
|
|
|
|
|
ur show subclasses |
60
|
|
|
|
|
|
|
ur show subclasses |
61
|
|
|
|
|
|
|
|
62
|
|
|
|
|
|
|
EOP |
63
|
0
|
|
|
|
|
|
return $result; |
64
|
|
|
|
|
|
|
} |
65
|
|
|
|
|
|
|
|
66
|
|
|
|
|
|
|
sub _mine_tree_for_class { |
67
|
0
|
|
|
0
|
|
|
my ($tree, $name, $result) = @_; |
68
|
|
|
|
|
|
|
|
69
|
0
|
0
|
|
|
|
|
if(ref($tree) eq 'HASH') { |
|
|
0
|
|
|
|
|
|
70
|
0
|
|
|
|
|
|
for my $key (keys %{$tree}) { |
|
0
|
|
|
|
|
|
|
71
|
0
|
0
|
|
|
|
|
if($key eq $name) { |
72
|
0
|
|
|
|
|
|
push(@{$result}, 1); |
|
0
|
|
|
|
|
|
|
73
|
|
|
|
|
|
|
} else { |
74
|
0
|
|
|
|
|
|
_mine_tree_for_class($tree->{$key}, $name, $result); |
75
|
|
|
|
|
|
|
} |
76
|
|
|
|
|
|
|
} |
77
|
|
|
|
|
|
|
} elsif(ref($tree) eq 'ARRAY') { |
78
|
0
|
|
|
|
|
|
for my $item (@{$tree}) { |
|
0
|
|
|
|
|
|
|
79
|
0
|
0
|
|
|
|
|
if($item eq $name) { |
80
|
0
|
|
|
|
|
|
push(@{$result}, 1); |
|
0
|
|
|
|
|
|
|
81
|
|
|
|
|
|
|
} |
82
|
0
|
|
|
|
|
|
_mine_tree_for_class($item, $name, $result); |
83
|
|
|
|
|
|
|
} |
84
|
|
|
|
|
|
|
} |
85
|
0
|
|
|
|
|
|
return; |
86
|
|
|
|
|
|
|
} |
87
|
|
|
|
|
|
|
|
88
|
|
|
|
|
|
|
sub execute { |
89
|
0
|
|
|
0
|
|
|
my ($self) = @_; |
90
|
0
|
|
|
|
|
|
my $indexfile = '/tmp/.ur_class_index'; |
91
|
|
|
|
|
|
|
|
92
|
0
|
|
|
|
|
|
my $subclass_index_ref; |
93
|
0
|
0
|
0
|
|
|
|
if($self->recalculate or (not -e $indexfile)) { |
94
|
0
|
|
|
|
|
|
my $test_use_cmd = UR::Namespace::Command::Test::Use->create(); |
95
|
0
|
|
|
|
|
|
$test_use_cmd->execute(); |
96
|
|
|
|
|
|
|
|
97
|
0
|
|
|
|
|
|
$subclass_index_ref = {}; |
98
|
0
|
|
|
|
|
|
create_subclass_index('UR::Object', $subclass_index_ref); |
99
|
|
|
|
|
|
|
|
100
|
0
|
|
|
|
|
|
my %subclass_index = %{$subclass_index_ref}; |
|
0
|
|
|
|
|
|
|
101
|
0
|
|
|
|
|
|
open(my $output_fh, '>', $indexfile); |
102
|
0
|
|
|
|
|
|
for my $key (keys %subclass_index) { |
103
|
|
|
|
|
|
|
print $output_fh sprintf("%s %s\n", $key, |
104
|
0
|
|
|
|
|
|
join("\t", @{$subclass_index{$key}})); |
|
0
|
|
|
|
|
|
|
105
|
|
|
|
|
|
|
} |
106
|
0
|
|
|
|
|
|
close($output_fh); |
107
|
|
|
|
|
|
|
} else { |
108
|
0
|
|
|
|
|
|
$subclass_index_ref = parse_subclass_index_file($indexfile); |
109
|
|
|
|
|
|
|
} |
110
|
|
|
|
|
|
|
|
111
|
|
|
|
|
|
|
# check to see if superclass is even in the subclass_index |
112
|
0
|
|
|
|
|
|
my @result; |
113
|
0
|
|
|
|
|
|
_mine_tree_for_class($subclass_index_ref, $self->superclass, \@result); |
114
|
0
|
0
|
|
|
|
|
unless(@result) { |
115
|
0
|
0
|
|
|
|
|
my $class_name = $self->color ? |
116
|
|
|
|
|
|
|
Term::ANSIColor::colored($self->superclass, 'red') : |
117
|
|
|
|
|
|
|
$self->superclass; |
118
|
0
|
|
|
|
|
|
printf "%s is not a valid class, check your spelling or " . |
119
|
|
|
|
|
|
|
"see --help (recalculate).\n", $class_name; |
120
|
0
|
|
|
|
|
|
return; |
121
|
|
|
|
|
|
|
} |
122
|
|
|
|
|
|
|
|
123
|
0
|
0
|
|
|
|
|
if($self->flat) { |
124
|
0
|
|
|
|
|
|
$self->display_subclasses_flat($subclass_index_ref, |
125
|
|
|
|
|
|
|
$self->superclass, 0) |
126
|
|
|
|
|
|
|
} else { |
127
|
0
|
|
|
|
|
|
$self->display_subclasses($subclass_index_ref, |
128
|
|
|
|
|
|
|
$self->superclass, '', ' ', 0); |
129
|
|
|
|
|
|
|
} |
130
|
|
|
|
|
|
|
|
131
|
0
|
|
|
|
|
|
return 1; |
132
|
|
|
|
|
|
|
} |
133
|
|
|
|
|
|
|
|
134
|
|
|
|
|
|
|
sub create_subclass_index { |
135
|
0
|
|
|
0
|
0
|
|
my ($seed, $index_ref) = @_; |
136
|
|
|
|
|
|
|
|
137
|
0
|
|
|
|
|
|
my @children = $seed->__meta__->subclasses_loaded; |
138
|
0
|
|
|
|
|
|
for my $child (@children) { |
139
|
0
|
|
|
|
|
|
my @parents = @{$child->__meta__->{is}}; |
|
0
|
|
|
|
|
|
|
140
|
0
|
|
|
|
|
|
for my $parent (@parents) { |
141
|
0
|
0
|
|
|
|
|
if($index_ref->{$parent}) { |
142
|
0
|
|
|
|
|
|
push(@{$index_ref->{$parent}}, $child); |
|
0
|
|
|
|
|
|
|
143
|
|
|
|
|
|
|
} else { |
144
|
0
|
|
|
|
|
|
$index_ref->{$parent} = [$child]; |
145
|
|
|
|
|
|
|
} |
146
|
|
|
|
|
|
|
} |
147
|
|
|
|
|
|
|
} |
148
|
|
|
|
|
|
|
} |
149
|
|
|
|
|
|
|
|
150
|
|
|
|
|
|
|
sub parse_subclass_index_file { |
151
|
0
|
|
|
0
|
0
|
|
my ($indexfile) = @_; |
152
|
|
|
|
|
|
|
|
153
|
0
|
|
|
|
|
|
open(IN, '<', $indexfile); |
154
|
0
|
|
|
|
|
|
my %index; |
155
|
0
|
|
|
|
|
|
while(my $line = ) { |
156
|
0
|
|
|
|
|
|
chomp($line); |
157
|
0
|
0
|
|
|
|
|
if($line) { |
158
|
0
|
|
|
|
|
|
my ($parent, $rest) = split(/ /, $line); |
159
|
0
|
0
|
|
|
|
|
if($rest) { |
160
|
0
|
|
|
|
|
|
my @children = split('\t', $rest); |
161
|
0
|
|
|
|
|
|
$index{$parent} = \@children; |
162
|
|
|
|
|
|
|
} else { |
163
|
0
|
|
|
|
|
|
$index{$parent} = []; |
164
|
|
|
|
|
|
|
} |
165
|
|
|
|
|
|
|
} |
166
|
|
|
|
|
|
|
} |
167
|
0
|
|
|
|
|
|
return \%index |
168
|
|
|
|
|
|
|
} |
169
|
|
|
|
|
|
|
|
170
|
|
|
|
|
|
|
sub display_subclasses_flat { |
171
|
0
|
|
|
0
|
0
|
|
my ($self, $index_ref, $name, $depth) = @_; |
172
|
0
|
|
|
|
|
|
my $maximum_depth = $self->maximum_depth; |
173
|
0
|
0
|
0
|
|
|
|
if($depth == $maximum_depth + 1 and $maximum_depth != -1) { |
174
|
0
|
|
|
|
|
|
return; |
175
|
|
|
|
|
|
|
} |
176
|
0
|
|
|
|
|
|
print "$name\n"; |
177
|
|
|
|
|
|
|
|
178
|
|
|
|
|
|
|
# get the children |
179
|
0
|
|
|
|
|
|
my $children_ref = $index_ref->{$name}; |
180
|
0
|
|
|
|
|
|
my @children; |
181
|
0
|
0
|
|
|
|
|
if($children_ref) { |
182
|
0
|
|
|
|
|
|
@children = @{$index_ref->{$name}}; |
|
0
|
|
|
|
|
|
|
183
|
|
|
|
|
|
|
} else { # if it isn't in index it has no children. |
184
|
0
|
|
|
|
|
|
@children = (); |
185
|
|
|
|
|
|
|
} |
186
|
|
|
|
|
|
|
|
187
|
|
|
|
|
|
|
# loop over children |
188
|
0
|
|
|
|
|
|
for my $child (@children) { |
189
|
0
|
|
|
|
|
|
$self->display_subclasses_flat($index_ref, $child, $depth+1); |
190
|
|
|
|
|
|
|
} |
191
|
|
|
|
|
|
|
} |
192
|
|
|
|
|
|
|
|
193
|
|
|
|
|
|
|
sub display_subclasses { |
194
|
0
|
|
|
0
|
0
|
|
my ($self, $index_ref, $name, $global_prefix, $personal_prefix, $depth) = @_; |
195
|
0
|
|
|
|
|
|
my $maximum_depth = $self->maximum_depth; |
196
|
|
|
|
|
|
|
|
197
|
0
|
|
|
|
|
|
my ($dgp, $dpp, $dn) = ($global_prefix, $personal_prefix, $name); |
198
|
0
|
0
|
|
|
|
|
if($self->color) { |
199
|
0
|
|
|
|
|
|
($dgp, $dpp, $dn) = colorize_output($global_prefix, $personal_prefix, |
200
|
|
|
|
|
|
|
$name, $self->superclass); |
201
|
|
|
|
|
|
|
} |
202
|
0
|
|
|
|
|
|
print join('', $dgp, $dpp, $spacing, $dn); |
203
|
|
|
|
|
|
|
|
204
|
0
|
0
|
|
|
|
|
my $o = ($personal_prefix =~ /^\|/ ) ? '|' : ' '; |
205
|
0
|
|
|
|
|
|
my $child_global_prefix = sprintf("%s%s %s", $global_prefix, $o, $spacing); |
206
|
|
|
|
|
|
|
|
207
|
|
|
|
|
|
|
# get the children |
208
|
0
|
|
|
|
|
|
my $children_ref = $index_ref->{$name}; |
209
|
0
|
|
|
|
|
|
my @children; |
210
|
0
|
0
|
|
|
|
|
if($children_ref) { |
211
|
0
|
|
|
|
|
|
@children = @{$index_ref->{$name}}; |
|
0
|
|
|
|
|
|
|
212
|
|
|
|
|
|
|
} else { # if it isn't in index it has no children. |
213
|
0
|
|
|
|
|
|
@children = (); |
214
|
|
|
|
|
|
|
} |
215
|
|
|
|
|
|
|
|
216
|
|
|
|
|
|
|
# loop over children |
217
|
0
|
|
|
|
|
|
my $len_children = scalar(@children); |
218
|
0
|
0
|
0
|
|
|
|
if($len_children and $depth == $maximum_depth and $maximum_depth != -1) { |
|
|
|
0
|
|
|
|
|
219
|
0
|
|
|
|
|
|
print " ...\n"; |
220
|
0
|
|
|
|
|
|
return; |
221
|
|
|
|
|
|
|
} |
222
|
0
|
|
|
|
|
|
print "\n"; |
223
|
|
|
|
|
|
|
|
224
|
0
|
|
|
|
|
|
my $i = 1; |
225
|
0
|
|
|
|
|
|
for my $child (@children) { |
226
|
0
|
0
|
|
|
|
|
my $child_personal_prefix = ($len_children == $i) ? '`-' : '|-'; |
227
|
|
|
|
|
|
|
|
228
|
0
|
|
|
|
|
|
$self->display_subclasses($index_ref, $child, $child_global_prefix, |
229
|
|
|
|
|
|
|
$child_personal_prefix, $depth+1); |
230
|
0
|
|
|
|
|
|
$i += 1; |
231
|
|
|
|
|
|
|
} |
232
|
|
|
|
|
|
|
} |
233
|
|
|
|
|
|
|
|
234
|
|
|
|
|
|
|
sub colorize_output { |
235
|
0
|
|
|
0
|
0
|
|
my ($global_prefix, $personal_prefix, $name, $superclass) = @_; |
236
|
|
|
|
|
|
|
|
237
|
0
|
|
|
|
|
|
my $dgp = Term::ANSIColor::colored($global_prefix, 'white'); |
238
|
0
|
|
|
|
|
|
my $dpp = Term::ANSIColor::colored($personal_prefix, 'white'); |
239
|
0
|
|
|
|
|
|
my $name_prefix = $name; |
240
|
0
|
0
|
|
|
|
|
if($name_prefix =~ /^($superclass)/) { |
241
|
0
|
|
|
|
|
|
$name_prefix = $superclass; |
242
|
|
|
|
|
|
|
} else { |
243
|
0
|
|
|
|
|
|
$name_prefix = ''; |
244
|
|
|
|
|
|
|
} |
245
|
0
|
|
|
|
|
|
my $name_suffix = $name; |
246
|
0
|
|
|
|
|
|
$name_suffix =~ s/^($superclass)//; |
247
|
0
|
|
|
|
|
|
my $dn = sprintf("%s%s", Term::ANSIColor::colored($name_prefix, 'white'), $name_suffix ); |
248
|
|
|
|
|
|
|
|
249
|
0
|
|
|
|
|
|
return ($dgp, $dpp, $dn); |
250
|
|
|
|
|
|
|
} |
251
|
|
|
|
|
|
|
|
252
|
|
|
|
|
|
|
1; |