1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222
|
package Pod::Pandoc::Modules;
use strict;
use warnings;
use 5.010;
our $VERSION = '0.30';
use File::Path qw(make_path);
use File::Find;
use File::Spec;
use File::stat;
use File::Basename qw(dirname);
use Pandoc::Elements;
use Carp qw(croak);
sub new {
bless( ( $_[1] // {} ), $_[0] );
}
sub add {
my ( $self, $name, $doc ) = @_;
if ( my $given = $self->{$name} ) {
return if $doc->metavalue('file') !~ /\.pod/;
}
$doc->meta->{title} //= MetaString $name;
$self->{$name} = $doc;
}
sub module_link {
my ( $module, $opt ) = @_;
my $target = $module;
if ( $opt->{subdirectories} ) { # TODO: document and test
$target =~ s{::}{/}g;
}
else {
$target =~ s{::}{-}g;
}
if ( $opt->{wiki} ) {
return Link attributes {}, [ Str $module ], [ $target, 'wikilink' ];
}
else {
$target .= '.' . ( $opt->{ext} // 'html' );
return Link attributes {}, [ Str $module ], [ $target, $module ];
}
}
sub index {
my ( $modules, %opt ) = @_;
# TODO: extend, document, and test metadata
my %meta = map { $_ => MetaString "" . $opt{$_} }
grep { defined $opt{$_} } qw{title};
my @definitions = map {
[
[ module_link( $_, \%opt ) ],
[
[
Plain [ Str( $modules->{$_}->metavalue('subtitle') // '' ) ]
]
]
]
} sort keys %$modules;
Document \%meta, [ DefinitionList \@definitions ];
}
sub serialize {
my ( $modules, $dir, $opt, @args ) = _parse_arguments(@_);
# adjust links
# TODO: create copy instead of transforming, so
# this can be called multiple times!
foreach my $doc ( values %$modules ) {
$doc->transform(
Link => sub {
# TODO: use configured prefix instead of hard-coded URL base
my ( $module, $hash ) =
$_->url =~ qr{^https://metacpan\.org/pod/([^#]+)(#.*)?$};
return unless ( $module and $modules->{$module} );
# TODO: check whether hash link target exists
my $link = module_link( $module, $opt );
if ( defined $hash ) {
$link->url( $link->url . $hash );
}
return $link;
}
);
}
# serialize
foreach my $doc ( values %$modules ) {
my $file = $doc->metavalue('file');
my $module = $doc->metavalue('title');
my $name = $module;
if ( $opt->{subdirectories} ) {
$name =~ s{::}{/}g;
}
else {
$name =~ s{::}{-}g;
}
$name .= '.' . ( $opt->{ext} // 'html' );
my $target = File::Spec->catfile( $dir, $name );
if ( $opt->{update} and -e $target ) {
next if stat($file)->[9] <= stat($target)->[9];
}
make_path( dirname($target) );
$doc->to_pandoc( -o => $target, @args );
say "$file => $target" unless $opt->{quiet};
}
# create index file
if ( $opt->{index} ) {
my $index = $modules->index(%$opt);
my $target =
File::Spec->catfile( $dir, $opt->{index} . '.' . $opt->{ext} );
$index->to_pandoc( @args, -o => $target );
say $target unless $opt->{quiet};
}
}
sub _parse_arguments {
my $modules = shift;
my $dir = ref $_[0] ? undef : shift;
my %opt = ref $_[0] ? %{ shift() } : ();
$dir //= $opt{dir} // croak "output directory must be specified!";
$opt{index} = 'index' unless exists $opt{index};
$opt{ext} //= 'html';
$opt{ext} =~ s/^\.//;
croak "ext must not be .pm or .pod" if $opt{ext} =~ /^(pod|pm)$/;
( $modules, $dir, \%opt, @_ );
}
1;
__END__
=head1 NAME
Pod::Pandoc::Modules - set of parsed documentation of Perl modules
=head1 SYNOPSIS
use Pod::Simple::Pandoc;
my $modules = Pod::Simple::Pandoc->new->parse_modules('lib');
$modules->serialize( { target => 'doc' }, '--template' => '...' ] ); # TODO
=head1 DESCRIPTION
Module to serialize Pod from a set of parsed Perl or Pod files. Can be
configured via templates, document rewriting etc. and used with many output
formats (html, markdown, and rst to be embedded in static site generators such
as Jekyll).
See L<Pod::Simple::Pandoc> for how to create instances of this module.
=head1 METHODS
=head2 add( $name => $doc )
Add a module given as L<Pandoc::Document> unless a module of same C<$name>
already exists. As an exception a parsed L<.pod> file will override existing
entries. The document title is set to the module name if missing.
=head2 serialize ( [ $dir ] [, \%options ] [, @args ] )
Serialize a set of modules into a given directory.
This method is experimental and may change!
=over
=item dir
Output directory.
=item ext
Output file extension. Set to the value of C<format> by default.
=item index
Index filename (with or without extension). Set to C<index> by default. Use a
false value to disable index generation.
=item wiki
Don't create subdirectories and use wiki links for references between files.
instead.
=item update
Generate target files even if source files have not been updated.
=item quiet
Don't emit warnings and status information.
=back
=head2 index ( %options )
Create and return an index document as L<Pandoc::Document>.
=head1 SEE ALSO
This module is part of L<Pod::Pandoc>.
=cut
|