sub
view_seq_bold {
$_
[1] }
sub
view_seq_italic {
$_
[1] }
sub
view_seq_code {
$_
[1] }
sub
view_seq_file {
$_
[1] }
sub
view_verbatim {
$_
[1] }
sub
view_seq_link {
my
(
$self
,
$link
) =
@_
;
$link
=~ s/^.*?\|//;
return
$link
;
}
1;
our
$DEBUG
= 0;
BEGIN {
delete
$Pod::POM::Node::
{error};
}
sub
Pod::POM::Node::error {
my
(
$self
,
@rest
) =
@_
;
print
STDERR Carp::longmess;
die
"->error on Pod::POM::Node: @rest"
;
}
our
$VERSION
=
'0.91'
;
sub
new {
my
(
$class
,
$args
) =
@_
;
$args
= {
skip_underscored
=> 1,
input_files
=> [],
out_dir
=>
''
,
class_map
=> {},
skip_classes
=> [],
skip_inherits
=> [],
force_inherits
=> {},
method_format
=>
'%m'
,
%{
$args
|| {} },
};
$DEBUG
=
$args
->{debug} || 0;
if
(
$DEBUG
>= 2) {
Data::Dump::Streamer->
import
(
'Dump'
);
}
for
(
qw/input_files skip_classes skip_inherits/
) {
$args
->{
$_
} = [
$args
->{
$_
}]
if
not
ref
(
$args
->{
$_
}) eq
'ARRAY'
;
}
if
(
my
$fi
=
$args
->{force_inherits}) {
for
(
keys
%$fi
) {
$fi
->{
$_
} = [
$fi
->{
$_
}]
if
not
ref
(
$fi
->{
$_
}) eq
'ARRAY'
;
}
}
my
$self
=
bless
(
$args
,
$class
);
@{
$self
->{skip_classes}} =
grep
{
ref
}
map
{
$self
->_any_to_type_array(
$_
, 0,
'skip_classes'
); } @{
$self
->{skip_classes}};
if
(
my
$fi
=
$self
->{force_inherits}) {
$self
->{force_inherits_type} = {};
my
@fi_keys
=
keys
%$fi
;
foreach
my
$dest_doc
(
@fi_keys
) {
my
$type_any
=
$self
->_any_to_type_array(
$dest_doc
, 1,
'force_inherits keys'
);
unless
(
$type_any
) {
delete
$fi
->{
$dest_doc
};
next
;
}
my
(
$type
,
$any
) =
@$type_any
;
$self
->{force_inherits_type}{
$any
} =
$type
;
if
(
$dest_doc
ne
$any
) {
$fi
->{
$any
} =
$fi
->{
$any
} ? [ @{
$fi
->{
$any
}}, @{
$fi
->{
$dest_doc
}} ] :
$fi
->{
$dest_doc
};
delete
$fi
->{
$dest_doc
};
}
}
}
return
$self
;
}
sub
write_pod {
my
(
$self
) =
@_
;
my
(
$fi
,
$fit
) = (
$self
->{force_inherits},
$self
->{force_inherits_type});
my
@targets
=
map
{
-d
$_
? [
$_
,
$_
] : [
$_
, Path::Class::File->new(
$_
)->dir]
} @{
$self
->{input_files} };
die
"no targets"
if
(!
@targets
);
while
(
@targets
) {
my
(
$target
,
$origtarget
) = @{
shift
@targets
};
print
"target=$target origtarget=$origtarget \n"
if
(
$DEBUG
);
my
$filename
= (-d
$target
? Path::Class::Dir->new(
$target
) : Path::Class::File->new(
$target
))->cleanup->resolve;
my
$classname
=
$self
->_pure_filename_to_classname(
$filename
->relative(
$origtarget
) );
if
(
my
$skipped
= first {
$self
->_match_filename_to_type_array(
$classname
,
$filename
,
$_
); } @{
$self
->{skip_classes}} ) {
print
" target skipped per skip_classes: "
.(
ref
$skipped
?
$skipped
->[1] :
$skipped
).
"\n"
if
(
$DEBUG
);
next
;
}
if
(-d
$target
) {
print
" directory: adding children as new targets\n"
if
(
$DEBUG
);
unshift
@targets
,
map
{ [
$_
,
$origtarget
] } (
$filename
->children);
next
;
}
my
$should_process
= 0;
$should_process
= 1
if
(
$target
=~ m/\.pm$/);
if
(
$target
=~ m/\.pod$/) {
print
" POD: found\n"
if
(
$DEBUG
);
if
(
my
$forced
= first {
$self
->_match_filename_to_type_array(
$classname
,
$filename
, [
$fit
->{
$_
},
$_
]); }
keys
%$fi
) {
print
" POD: processing due to force_inherits match: $forced\n"
if
(
$DEBUG
);
$should_process
= 1;
}
}
if
(
$should_process
) {
my
$output_filename
=
$self
->{out_dir} ?
$filename
->relative(
$origtarget
)->absolute(
$self
->{out_dir}) :
$filename
;
$output_filename
=~ s/\.pm$/.pod/;
$output_filename
= Path::Class::File->new(
$output_filename
);
if
(
$self
->_is_ours(
$output_filename
)) {
my
$allpod
=
$self
->create_pod(
$target
,
$origtarget
);
if
(!
$allpod
) {
print
" not creating empty file $output_filename\n"
if
(
$DEBUG
);
next
;
}
my
$dir
=
$output_filename
->dir;
my
$ret
=
$dir
->mkpath;
my
(
$outfh
,
$oldperm
);
print
" Writing $output_filename\n"
if
(
$DEBUG
);
unless
(
$outfh
=
$output_filename
->
open
(
'w'
) ) {
if
($!{EACCES} and
$self
->{force_permissions} ) {
$output_filename
->remove;
$oldperm
=
$dir
->
stat
->mode;
chmod
$oldperm
| 0200,
$dir
or
die
"Can't chmod "
.
$dir
.
" (or write into it)"
;
$outfh
=
$output_filename
->
open
(
'w'
) or
die
"Can't open $output_filename for output (even after chmodding it's parent directory): $!"
;
}
else
{
die
"Can't open $output_filename for output: $!"
;
}
}
$outfh
->
print
(
$allpod
);
$outfh
->
close
;
if
(
defined
$oldperm
) {
chmod
$oldperm
,
$dir
or
die
sprintf
"Can't chmod %s back to 0%o"
,
$dir
,
$oldperm
;
}
}
}
}
return
1;
}
sub
create_pod {
my
(
$self
,
$src
,
$root_dir
) =
@_
;
my
$class_map
=
$self
->{class_map};
die
"create_pod needs a source file argument!"
unless
(
$src
);
$src
= Path::Class::File->new(
$src
)->cleanup->resolve;
my
(
$fi
,
$fit
) = (
$self
->{force_inherits},
$self
->{force_inherits_type});
my
(
$tt_stash
,
$classname
,
@isa_flattened
);
unless
(
$src
=~ m/\.pod$/) {
$classname
=
$tt_stash
->{classname} =
$self
->_require_class(
$src
) ||
return
;
@isa_flattened
= @{mro::get_linear_isa(
$classname
)};
}
else
{
$classname
=
$tt_stash
->{classname} =
$self
->_pure_filename_to_classname(
$root_dir
?
$src
->relative(
$root_dir
) :
$src
);
$self
->_check_pod_sections(
$src
,
$classname
);
}
my
$force_inherits
= (first {
$self
->_match_filename_to_type_array(
$classname
,
$src
, [
$fit
->{
$_
},
$_
]); }
keys
%$fi
) ||
''
;
$force_inherits
=
$fi
->{
$force_inherits
};
if
(
$force_inherits
) {
foreach
my
$class
(
@$force_inherits
) {
print
" Found force inherit: $class\n"
if
(
$DEBUG
);
$self
->_require_class(
undef
,
$class
) ||
return
;
push
@isa_flattened
, @{mro::get_linear_isa(
$class
)};
}
}
foreach
my
$s
( @{
$self
->{skip_inherits} },
$classname
) {
for
(
my
$i
= 0;
$i
<
@isa_flattened
;
$i
++) {
if
(
$s
eq
$isa_flattened
[
$i
]) {
print
" Skipped per skip_inherits: $s\n"
if
(
$DEBUG
);
splice
(
@isa_flattened
,
$i
--, 1);
}
}
}
if
(!
@isa_flattened
) {
print
" No parent classes\n"
if
(
$DEBUG
);
return
;
}
$tt_stash
->{isa_flattened} = \
@isa_flattened
;
if
(
exists
$self
->{dead_links}) {
foreach
my
$class
(
@isa_flattened
) {
$self
->_check_pod_sections(
undef
,
$class
);
}
}
my
%seen
;
for
my
$parent_class
(
@isa_flattened
) {
print
" Parent class: $parent_class\n"
if
(
$DEBUG
);
my
$stash
;
{
no
strict
'refs'
;
$stash
= \%{
"$parent_class\::"
};
}
my
$local_config
=
$stash
->{_pod_inherit_config};
if
(not
exists
$local_config
->{skip_underscored}) {
$local_config
->{skip_underscored} =
$self
->{skip_underscored};
}
$local_config
->{class_map} ||=
$class_map
;
for
my
$globname
(
sort
keys
%$stash
) {
next
if
(
$local_config
->{skip_underscored} and
$globname
=~ m/^_/);
next
if
$seen
{
$globname
};
next
if
(
$globname
=~ m/^(?:AUTOLOAD|CLONE|DESTROY|BEGIN|UNITCHECK|CHECK|INIT|END)$/);
my
$glob
=
$stash
->{
$globname
};
my
$exists
;
eval
{
$exists
=
exists
&$glob
;
};
if
($@) {
if
($@ =~ /Not a subroutine reference/) {
print
" Got not a subref for $globname in $parent_class; it is probably imported accidentally.\n"
if
(
$DEBUG
);
$exists
=0;
}
else
{
die
"While checking if $parent_class $globname is a sub: $@"
;
}
}
next
unless
(
$exists
);
my
$nice_name
;
if
(
$globname
eq
'()'
) {
$nice_name
=
'I<overload table>'
;
}
elsif
(
$globname
=~ m/^\((.*)/) {
my
$sort
= $1;
$sort
=~ s/(.)/
sprintf
"E<%d>"
,
ord
$1/ge;
$nice_name
=
"I<$sort overloading>"
;
}
else
{
$nice_name
=
$globname
;
}
my
$subref
=
$classname
->can(
$globname
);
if
(
$force_inherits
&& !
$subref
) {
foreach
my
$class
(
@$force_inherits
) {
$subref
=
$class
->can(
$globname
)
unless
defined
$subref
;
}
}
next
if
!
$subref
;
my
$identify_name
= Sub::Identify::stash_name(
$subref
);
next
if
$identify_name
eq
'UNIVERSAL'
;
if
(
$identify_name
ne
$parent_class
) {
next
;
}
$seen
{
$globname
} =
$parent_class
;
my
$doc_parent_class
=
$local_config
->{class_map}->{
$parent_class
} ||
$parent_class
;
if
(
exists
$self
->{dead_links}) {
my
$found_doc
= 0;
foreach
my
$class
(
$parent_class
,
@isa_flattened
, @{mro::get_linear_isa(
$parent_class
)}) {
next
if
(first {
$_
eq
$class
} @{
$self
->{skip_inherits} });
my
$map_class
=
$local_config
->{class_map}->{
$class
} ||
$class
;
$self
->_check_pod_sections(
undef
,
$map_class
);
if
(
$self
->{pod_sections}{
$map_class
}{
$globname
}) {
print
" Method documentation on grandparent: $map_class"
.
"::$globname\n"
if
(
$DEBUG
&&
$doc_parent_class
ne
$map_class
);
$doc_parent_class
=
$map_class
;
$found_doc
= 1;
last
;
}
}
if
(
$self
->{dead_links} eq
''
&& !
$found_doc
) {
print
" Skipped due to lack of documentation: $globname\n"
if
(
$DEBUG
);
next
;
}
}
push
@{
$tt_stash
->{methods}{
$doc_parent_class
}},
$nice_name
;
splice
(
@isa_flattened
, (firstidx {
$_
eq
$parent_class
}
@isa_flattened
), 0,
$doc_parent_class
)
unless
(any {
$_
eq
$doc_parent_class
}
@isa_flattened
);
}
}
return
if
!
keys
%{
$tt_stash
->{methods}};
my
$new_pod
=
<<'__END_POD__';
=head1 INHERITED METHODS
=over
__END_POD__
$new_pod
=~ s/^ //mg;
for
my
$class
(@{
$tt_stash
->{isa_flattened}}) {
next
unless
(
$tt_stash
->{methods}{
$class
});
$new_pod
.=
"=item L<$class>\n\n"
;
$new_pod
.=
join
(
", "
,
map
{
my
$method
=
$_
;
my
$mlf
= (
exists
$self
->{dead_links} &&
$self
->{dead_links} ne
''
&& !
$self
->{pod_sections}{
$class
}{
$method
}) ?
$self
->{dead_links} :
$self
->{method_format};
$mlf
=~ s/\
%m
/
$method
/g;
$mlf
=~ s/\
%c
/
$class
/g;
$mlf
=~ s/\%\%/\%/g;
$mlf
;
} @{
$tt_stash
->{methods}{
$class
}}) .
"\n\n"
;
}
$new_pod
.=
"=back\n\n=cut\n\n"
;
print
"New pod, before Pod::POMification: \n"
,
$new_pod
if
(
$DEBUG
>= 2);
my
$parser
= Pod::POM->new;
$new_pod
=
$parser
->parse_text(
$new_pod
)
or
die
"Generated pod invalid?"
;
foreach
my
$warning
(
$parser
->warnings()) {
warn
"Generated pod warning: $warning\n"
;
}
if
(
$DEBUG
>= 2) {
print
"New pod, after Pod::POMification: \n"
;
print
$new_pod
->
dump
;
}
$parser
= Pod::POM->new;
my
$pod
=
$parser
->parse_file(
$src
->stringify)
or
die
"Couldn't parse existing pod in $src: "
.
$parser
->error;
my
$outstr
=
$self
->_get_inherit_header(
$classname
,
$src
);
my
$before
;
my
$insertion_point
;
my
$i
= 0;
for
(
reverse
$pod
->content) {
$i
--;
next
unless
$_
->isa(
'Pod::POM::Node::Head1'
);
my
$title
=
$_
->title;
if
(
grep
{
$title
eq
$_
}
qw<LICENSE AUTHORS LIMITATIONS CONTRIBUTORS AUTHOR CAVEATS COPYRIGHT BUGS>
,
'SEE ALSO'
,
'ALSO SEE'
,
'WHERE TO GO NEXT'
,
'COPYRIGHT AND LICENSE'
) {
print
" Fount head $title at index $i, going before that section\n"
if
$DEBUG
;
$insertion_point
=
$i
;
$before
= 1;
last
;
}
else
{
print
" Found head $title at index $i, going after that section\n"
if
$DEBUG
;
$insertion_point
=
$i
;
$before
= 0;
last
;
}
}
if
(!
$insertion_point
and
$pod
->content) {
print
" Going at end\n"
if
$DEBUG
;
$insertion_point
= -1;
$before
= 0;
}
if
(!
$insertion_point
) {
print
" Going as only section\n"
if
$DEBUG
;
$insertion_point
=
$pod
;
$outstr
.=
$new_pod
;
return
$outstr
;
}
if
(not
$before
and
$insertion_point
== -1) {
push
@{
$pod
->{content}},
$new_pod
;
}
elsif
(
$before
) {
splice
(@{
$pod
->content},
$insertion_point
-1, 0,
$new_pod
);
}
else
{
splice
(@{
$pod
->content},
$insertion_point
, 0,
$new_pod
);
}
$outstr
.=
$pod
;
return
$outstr
;
}
sub
_file_to_package {
my
(
$self
,
$file
) =
@_
;
open
my
$fh
,
"<"
,
$file
or
die
"Can't open $file: $!"
;
while
(<
$fh
>) {
return
$1
if
(m/^
package
\s+([A-Za-z0-9_:]+);/);
if
(m/^
package
\b/) {
print
" Package hidden with anti-PAUSE tricks in $file\n"
if
(
$DEBUG
);
return
undef
;
}
}
print
" Couldn't find any package statement in $file\n"
if
(
$DEBUG
);
return
undef
;
}
sub
_pure_filename_to_classname {
my
(
$self
,
$pure_filename
) =
@_
;
$pure_filename
=~ s/\.p(?:m|od)$//i;
return
join
'::'
,
split
(/::|\/|\\/,
$pure_filename
);
}
sub
_any_to_pm_filename {
my
(
$self
,
$any
) =
@_
;
$any
=~ s/\.p(?:m|od)$//i;
return
Path::Class::File->new(
split
(/::|\/|\\/,
$any
.
'.pm'
) )->cleanup;
}
sub
_any_to_real_file {
my
(
$self
,
$any
,
$try_pods
,
$try_dirs
) =
@_
;
my
$filename
=
$self
->_any_to_pm_filename(
$any
);
foreach
my
$d
(@{
$self
->{input_files} },
'.'
) {
my
$pd
= -d
$d
?
$d
: Path::Class::File->new(
$d
)->dir;
my
$f
= Path::Class::File->new(
$pd
,
$filename
)->cleanup;
return
$f
->resolve
if
(-f
$f
);
next
unless
$try_pods
;
$f
=~ s/m$/od/;
return
Path::Class::File->new(
$f
)->resolve
if
(-f
$f
);
next
unless
$try_dirs
;
$f
=~ s/\.pod$//;
return
Path::Class::Dir->new(
$f
)->resolve
if
(-d
$f
);
}
return
undef
;
}
sub
_any_to_type_array {
my
(
$self
,
$any
,
$try_pods
,
$value_type
) =
@_
;
return
undef
unless
defined
$any
;
my
$type
;
$value_type
=
$value_type
?
"[Found in $value_type] "
:
''
;
my
$crossplat_any
= Path::Class::File->new(
split
(/\/|\\/,
$any
) )->cleanup->stringify;
my
$real_file
=
$self
->_any_to_real_file(
$any
,
$try_pods
, 1);
if
(
$any
=~ /::/) {
$type
=
'c'
; }
elsif
(
$any
=~ /\.p(?:m|od)$/i) {
$type
=
'f'
; }
elsif
(-d
$crossplat_any
) {
$type
=
'd'
; }
elsif
(-e
$crossplat_any
) {
$type
=
'f'
; }
elsif
(
$any
=~ /\/|\\/) {
unless
(
$real_file
) {
warn
$value_type
.
"Appears to be a file/dir, but it doesn't exist: $any"
;
return
undef
;
}
$type
= -d
$real_file
?
'd'
:
'f'
;
}
elsif
(
$real_file
) {
$type
=
'c'
; }
else
{
warn
$value_type
.
"Cannot even guess to what this is, as it doesn't exist anywhere: $any"
;
return
undef
;
}
return
[
$type
, (
$type
eq
'c'
) ?
$any
: (
$real_file
||
$crossplat_any
)];
}
sub
_match_filename_to_type_array {
my
(
$self
,
$classname
,
$full_filename
,
$type_any
) =
@_
;
$type_any
=
$self
->_any_to_type_array(
$type_any
)
unless
ref
$type_any
;
my
(
$type
,
$any
) =
@$type_any
;
return
$classname
eq
$any
if
(
$type
eq
'c'
);
return
$full_filename
eq
$any
if
(
$type
eq
'f'
);
return
$full_filename
=~ /^\Q
$any
\E/
if
(
$type
eq
'd'
);
return
undef
;
}
sub
_require_class {
my
(
$self
,
$src
,
$classname
) =
@_
;
$classname
||=
$self
->_file_to_package(
$src
) ||
return
undef
;
$src
||=
$self
->_any_to_real_file(
$classname
);
my
$class_as_filename
=
$self
->_any_to_pm_filename(
$classname
);
no
warnings
'redefine'
;
local
$|=1;
my
$old_sig_warn
=
$SIG
{__WARN__};
local
$SIG
{__WARN__} =
sub
{
return
if
(
$_
[0] =~ /^(?:Constant )?[Ss]ubroutine [\w\:]+ redefined /);
my
$warning
=
" While loading $src: "
.
$_
[0];
$old_sig_warn
?
$old_sig_warn
->(
$warning
) :
warn
$warning
;
};
unless
(
exists
$INC
{
$class_as_filename
}) {
print
"Still no source found for $classname; forced to use 'require'\n"
if
(
$DEBUG
&& !
$src
);
my
$did_it
=
$src
?
do
$src
: Class::Load::load_optional_class(
$classname
);
unless
(
$did_it
) {
my
$err
= $@;
$err
=~ s/ \(\
@INC
contains: .*\)//;
$SIG
{__WARN__} =
$old_sig_warn
;
warn
"Couldn't autogenerate documentation for $src: $err\n"
;
return
undef
;
}
}
$INC
{
$class_as_filename
} =
$src
unless
(
exists
$INC
{
$class_as_filename
});
$self
->_check_pod_sections(
$src
,
$classname
);
return
$classname
;
}
sub
_check_pod_sections {
my
(
$self
,
$src
,
$classname
) =
@_
;
return
0
unless
(
$classname
);
return
0
unless
(
exists
$self
->{dead_links} && not
$self
->{pod_sections}{
$classname
});
$src
||=
$INC
{
$self
->_any_to_pm_filename(
$classname
) } ||
$self
->_any_to_real_file(
$classname
, 1, 1) ||
return
0
;
my
$hash
=
$self
->{pod_sections}{
$classname
} = {};
my
$p
= Pod::POM->new;
my
$pom
=
$p
->parse_file(
"$src"
) ||
die
$p
->error();
$self
->_find_pod_headers(
$pom
,
$hash
);
if
(
$DEBUG
) {
print
" Found "
.
scalar
(
keys
%$hash
).
" POD sections in $classname:\n"
;
print
" "
.
join
(
', '
,
keys
%$hash
).
"\n"
;
}
return
1;
}
sub
_find_pod_headers {
my
(
$self
,
$top
,
$hash
) =
@_
;
$hash
->{
$top
->title->present(
'Pod::POM::View::TextStrip'
) } = 1
if
(
$top
->type =~ /head/i);
foreach
my
$item
(
$top
->content) {
$self
->_find_pod_headers(
$item
,
$hash
);
}
}
sub
_is_ours {
my
(
$self
,
$outfn
) =
@_
;
if
(-e
$outfn
) {
open
my
$outfh
,
'<'
,
$outfn
or
die
"Can't open pre-existing $outfn for reading: $!"
;
if
(<
$outfh
> ne
"=for comment POD_DERIVED_INDEX_GENERATED\n"
) {
warn
"$outfn already exists, and it doesn't look like we generated it. Skipping this file"
;
return
0;
}
}
return
1;
}
sub
_get_inherit_header {
my
(
$self
,
$classname
,
$src
) =
@_
;
$src
= Path::Class::File->new(
$src
)->as_foreign(
'Unix'
);
return
<<__END_HEADER__;
=for comment POD_DERIVED_INDEX_GENERATED
The following documentation is automatically generated. Please do not edit
this file, but rather the original, inline with $classname
at $src
(on the system that originally ran this).
If you do edit this file, and don't want your changes to be removed, make
sure you change the first line.
=cut
__END_HEADER__
}
1;