123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614 |
- #! /usr/bin/env perl
- # $Id$
- # pod2texi -- convert Pod to Texinfo.
- # Copyright 2012, 2013, 2014, 2015, 2016 Free Software Foundation, Inc.
- #
- # This program is free software; you can redistribute it and/or modify
- # it under the terms of the GNU General Public License as published by
- # the Free Software Foundation; either version 3 of the License,
- # or (at your option) any later version.
- #
- # This program is distributed in the hope that it will be useful,
- # but WITHOUT ANY WARRANTY; without even the implied warranty of
- # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- # GNU General Public License for more details.
- #
- # You should have received a copy of the GNU General Public License
- # along with this program. If not, see <http://www.gnu.org/licenses/>.
- #
- # Original author: Patrice Dumas <pertusus@free.fr>
- use strict;
- use Getopt::Long qw(GetOptions);
- # for dirname.
- use File::Basename;
- use File::Spec;
- Getopt::Long::Configure("gnu_getopt");
- #use Pod::Simple::Debug (4);
- BEGIN
- {
- # emulate -w
- $^W = 1;
- my ($real_command_name, $command_directory, $command_suffix)
- = fileparse($0, '.pl');
- my $datadir = '@datadir@';
- my $package = '@PACKAGE@';
- my $updir = File::Spec->updir();
- my $texinfolibdir;
- my $lib_dir;
- # in-source run
- if (($command_suffix eq '.pl' and !(defined($ENV{'TEXINFO_DEV_SOURCE'})
- and $ENV{'TEXINFO_DEV_SOURCE'} eq 0)) or $ENV{'TEXINFO_DEV_SOURCE'}) {
- my $srcdir = defined $ENV{'srcdir'} ? $ENV{'srcdir'} : $command_directory;
- $texinfolibdir = File::Spec->catdir($srcdir, $updir, 'tp');
- $lib_dir = File::Spec->catdir($texinfolibdir, 'maintain');
- unshift @INC, (File::Spec->catdir($srcdir, 'lib'), $texinfolibdir);
- } elsif ($datadir ne '@' .'datadir@' and $package ne '@' . 'PACKAGE@'
- and $datadir ne '') {
- $texinfolibdir = File::Spec->catdir($datadir, $package);
- # try to make package relocatable, will only work if standard relative paths
- # are used
- if (! -f File::Spec->catfile($texinfolibdir, 'Texinfo', 'Parser.pm')
- and -f File::Spec->catfile($command_directory, $updir, 'share',
- 'texinfo', 'Texinfo', 'Parser.pm')) {
- $texinfolibdir = File::Spec->catdir($command_directory, $updir,
- 'share', 'texinfo');
- }
- $lib_dir = $texinfolibdir;
- unshift @INC, (File::Spec->catdir($texinfolibdir, 'Pod-Simple-Texinfo'),
- $texinfolibdir);
- }
- # '@USE_EXTERNAL_LIBINTL @ and similar are substituted in the
- # makefile using values from configure
- if (defined($texinfolibdir)) {
- if ('@USE_EXTERNAL_LIBINTL@' ne 'yes') {
- unshift @INC, (File::Spec->catdir($lib_dir, 'lib', 'libintl-perl', 'lib'));
- }
- if ('@USE_EXTERNAL_EASTASIANWIDTH@' ne 'yes') {
- unshift @INC, (File::Spec->catdir($lib_dir, 'lib', 'Unicode-EastAsianWidth', 'lib'));
- }
- if ('@USE_EXTERNAL_UNIDECODE@' ne 'yes') {
- unshift @INC, (File::Spec->catdir($lib_dir, 'lib', 'Text-Unidecode', 'lib'));
- }
- }
- }
- use Pod::Simple::Texinfo;
- use Texinfo::Common;
- use Texinfo::Parser;
- use Texinfo::Transformations;
- {
- # A fake package to be able to use Pod::Simple::PullParser without generating
- # any output.
- package Pod::Simple::PullParserRun;
- use vars qw(@ISA);
- @ISA = ('Pod::Simple::PullParser');
- sub new
- {
- return shift->SUPER::new(@_);
- }
- sub run(){};
- }
- my ($real_command_name, $directories, $suffix) = fileparse($0);
- # placeholder for string translations, not used for now
- sub __($)
- {
- return $_[0];
- }
- sub pod2texi_help()
- {
- return __("Usage: pod2texi [OPTION]... POD...
- Translate Perl pod documentation file(s) to Texinfo. There are two
- basic modes of operation. First, by default, each pod is translated to
- a standalone Texinfo manual.
- Second, if C<--base-level> is set higher than 0, each pod is translated
- to a file suitable for C<\@include>, and one more file with all the
- C<\@include>s is generated, intended to be C<\@include>d in turn within
- a hand-written top-level file.
- Options:
- --appendix-sections use appendix-like sections.
- --base-level=NUM|NAME level of the head1 commands; default 0.
- --debug=NUM set debugging level.
- --help display this help and exit.
- --no-fill-section-gaps do not fill sectioning gaps.
- --no-section-nodes use anchors for sections instead of nodes.
- --output=NAME output to NAME for the first or main manual
- instead of standard output.
- --preamble=STR insert STR as beginning boilerplate.
- --subdir=NAME put files included in the main manual in NAME.
- --top top for the main manual.
- --unnumbered-sections use unumbered sections.
- --version display version information and exit.
- Email bug reports to bug-texinfo\@gnu.org,
- general questions and discussion to help-texinfo\@gnu.org.
- Texinfo home page: http://www.gnu.org/software/texinfo/\n");
- }
- my $base_level = 0;
- my $unnumbered_sections = 0;
- my $appendix_sections = 0;
- my $output = '-';
- my $top = 'top';
- my $preamble = undef;
- my $subdir;
- my $section_nodes = 1;
- my $fill_sectioning_gaps = 1;
- my $debug = 0;
- my $result_options = Getopt::Long::GetOptions (
- 'help|h' => sub { print pod2texi_help(); exit 0; },
- 'version|V' => sub {print "$real_command_name $Pod::Simple::Texinfo::VERSION\n\n";
- printf __("Copyright (C) %s Free Software Foundation, Inc.
- License GPLv3+: GNU GPL version 3 or later <http://gnu.org/licenses/gpl.html>
- This is free software: you are free to change and redistribute it.
- There is NO WARRANTY, to the extent permitted by law.\n"), "2016";
- exit 0;},
- 'base-level=s' => sub {
- if ($_[1] =~ /^[0-4]$/) {
- $base_level = $_[1];
- } elsif (defined($Texinfo::Common::command_structuring_level{$_[1]})) {
- $base_level = $Texinfo::Common::command_structuring_level{$_[1]};
- } else {
- die sprintf(__("%s: wrong argument for --base-level\n"),
- $real_command_name);
- }
- },
- 'unnumbered-sections!' => \$unnumbered_sections,
- 'appendix-sections!' => \$appendix_sections,
- 'output|o=s' => \$output,
- 'preamble=s' => \$preamble,
- 'subdir=s' => \$subdir,
- 'top=s' => \$top,
- 'section-nodes!' => \$section_nodes,
- 'fill-section-gaps!' => \$fill_sectioning_gaps,
- 'debug=i' => \$debug,
- );
- exit 1 if (!$result_options);
- if (defined($subdir)) {
- if (! -d $subdir) {
- if (!mkdir($subdir)) {
- die sprintf(__("%s: could not create directory %s: %s"),
- $real_command_name, $subdir, $!);
- }
- }
- }
- my $STDOUT_DOCU_NAME = 'stdout';
- my @manuals;
- my @all_manual_names;
- my @input_files = @ARGV;
- # use STDIN if not a tty, like makeinfo does
- @input_files = ('-') if (!scalar(@input_files) and !-t STDIN);
- die sprintf(__("%s: missing file argument;\n"), $real_command_name)
- .sprintf(__("try `%s --help' for more information\n"), $real_command_name)
- unless (scalar(@input_files) >= 1);
- my @processed_files;
- # First gather all the manual names
- if ($base_level > 0) {
- foreach my $file (@input_files) {
- # we don't want to read from STDIN, as the input read would be lost
- # same with named pipe and socket...
- # FIXME are there other file that have the same problem?
- next if ($file eq '-' or -p $file or -S $file);
- # not really used, only the manual name is used.
- my $parser = Pod::Simple::PullParserRun->new();
- $parser->parse_file($file);
- my $short_title = $parser->get_short_title();
- if (defined($short_title) and $short_title =~ m/\S/) {
- push @manuals, $short_title;
- push @all_manual_names, $short_title;
- #print STDERR "NEW MANUAL: $short_title\n";
- } else {
- if (!$parser->content_seen) {
- warn sprintf(__("%s: ignoring %s without content\n"),
- $real_command_name, $file);
- next;
- }
- push @all_manual_names, undef;
- }
- push @processed_files, $file;
- }
- } else {
- @processed_files = @input_files;
- }
- sub _fix_texinfo_tree($$$$;$)
- {
- my $self = shift;
- my $manual_texi = shift;
- my $section_nodes = shift;
- my $fill_gaps_in_sectioning = shift;
- my $do_master_menu = shift;
- my $parser = Texinfo::Parser::parser();
- my $tree = $parser->parse_texi_text($manual_texi);
- if ($fill_gaps_in_sectioning) {
- my ($added_sections, $added_nodes);
- ($tree->{'contents'}, $added_sections)
- = Texinfo::Transformations::fill_gaps_in_sectioning($tree);
- # there should already be nodes associated with other sections. Therefore
- # new nodes should only be created for the $added_sections.
- if ($section_nodes) {
- ($tree->{'contents'}, $added_nodes)
- = Texinfo::Transformations::insert_nodes_for_sectioning_commands($parser, $tree);
- if ($self and $self->texinfo_sectioning_base_level > 0) {
- # prepend the manual name
- foreach my $node (@$added_nodes) {
- # First remove the old normalized entry
- delete $parser->{'labels'}->{$node->{'extra'}->{'normalized'}};
- # now get the number
- my $node_texi = Texinfo::Convert::Texinfo::convert(
- {'contents' => $node->{'extra'}->{'node_content'}});
- # We could have kept the asis, too, it is kept when !section_nodes
- $node_texi =~ s/^\s*(\@asis\{\})?\s*//;
- # complete with manual name
- my $complete_node_name = $self->_node_name($node_texi);
- # now recreate node arg, similar with Texinfo::Transformations::_new_node
- my $tree = Texinfo::Parser::parse_texi_text(undef, $complete_node_name);
- my $node_arg = $node->{'args'}->[0];
- $node_arg->{'contents'} = $tree->{'contents'};
- push @{$node_arg->{'contents'}},
- {'type' => 'spaces_at_end', 'text' => "\n"};
- unshift @{$node_arg->{'contents'}},
- {'extra' => {'command' => $node},
- 'text' => ' ',
- 'type' => 'empty_spaces_after_command'};
- foreach my $content (@{$node_arg->{'contents'}}) {
- $content->{'parent'} = $node_arg;
- }
- # Last parse and register node
- my $parsed_node = Texinfo::Parser::_parse_node_manual($node_arg);
- #push @{$node->{'extra'}->{'nodes_manuals'}}, $parsed_node;
- @{$node->{'extra'}->{'nodes_manuals'}} = ($parsed_node);
- if (!Texinfo::Parser::_register_label($parser, $node, $parsed_node, undef)) {
- print STDERR "BUG: node not unique, register failed: $parsed_node->{'normalized'}\n";
- }
- }
- }
- }
- }
- my $structure = Texinfo::Structuring::sectioning_structure($parser, $tree);
- Texinfo::Transformations::complete_tree_nodes_menus($parser, $tree)
- if ($section_nodes);
- Texinfo::Transformations::regenerate_master_menu($parser) if ($do_master_menu);
- return ($parser, $tree);
- }
- sub _fix_texinfo_manual($$$$;$)
- {
- my $self = shift;
- my $manual_texi = shift;
- my $section_nodes = shift;
- my $fill_gaps_in_sectioning = shift;
- my $do_master_menu = shift;
- my ($parser, $tree) = _fix_texinfo_tree($self, $manual_texi, $section_nodes,
- $fill_gaps_in_sectioning, $do_master_menu);
- return Texinfo::Convert::Texinfo::convert($tree);
- }
- sub _do_top_node_menu($)
- {
- my $manual_texi = shift;
- my ($parser, $tree) = _fix_texinfo_tree(undef, $manual_texi, 1, 0, 1);
- my $labels = $parser->labels_information();
- my $top_node_menu = $labels->{'Top'}->{'menus'}->[0];
- if ($top_node_menu) {
- return Texinfo::Convert::Texinfo::convert($top_node_menu);
- } else {
- return '';
- }
- }
- my $file_nr = 0;
- # Full manual is collected to generate the top node menu, if $section_nodes
- my $full_manual = '';
- my @included;
- foreach my $file (@processed_files) {
- my $manual_texi = '';
- my $outfile;
- my $name = shift @all_manual_names;
- if ($base_level == 0 and !$file_nr) {
- $outfile = $output;
- } else {
- if (defined($name)) {
- $outfile = Pod::Simple::Texinfo::_pod_title_to_file_name($name);
- $outfile .= '.texi';
- } else {
- if ($file eq '-') {
- $outfile = $STDOUT_DOCU_NAME;
- } else {
- $outfile = $file;
- }
- if ($outfile =~ /\.(pm|pod)$/) {
- $outfile =~ s/\.(pm|pod)$/.texi/i;
- } else {
- $outfile .= '.texi';
- }
- }
- $outfile = File::Spec->catfile($subdir, $outfile)
- if (defined($subdir));
- }
- my $new = Pod::Simple::Texinfo->new();
- push @included, [$name, $outfile, $file] if ($base_level > 0);
- my $fh;
- if ($outfile eq '-') {
- $fh = *STDOUT;
- } else {
- open (OUT, ">$outfile") or die sprintf(__("%s: could not open %s for writing: %s\n"),
- $real_command_name, $outfile, $!);
- $fh = *OUT;
- }
- # FIXME should use =encoding
- binmode($fh, ':encoding(utf8)');
- $new->output_string(\$manual_texi);
- $new->texinfo_sectioning_base_level($base_level);
- if ($section_nodes) {
- $new->texinfo_section_nodes(1);
- }
- if ($unnumbered_sections) {
- $new->texinfo_sectioning_style('unnumbered');
- } elsif ($appendix_sections) {
- $new->texinfo_sectioning_style('appendix');
- }
- if ($base_level > 0 and @manuals) {
- $new->texinfo_internal_pod_manuals(\@manuals);
- }
-
- print STDERR "processing $file -> $outfile ($name)\n" if ($debug);
- $new->parse_file($file);
- if ($section_nodes or $fill_sectioning_gaps) {
- if ($debug > 4) {
- # print to a file
- open (DBGFILE, ">$outfile-dbg") or die sprintf(__("%s: could not open %s: %s\n"),
- $real_command_name, "$outfile-dbg", $!);
- binmode(DBGFILE, ':encoding(utf8)');
- print DBGFILE $manual_texi;
-
- }
- $manual_texi = _fix_texinfo_manual($new, $manual_texi, $section_nodes,
- $fill_sectioning_gaps);
- $full_manual .= $manual_texi if ($section_nodes);
- }
- print $fh $manual_texi;
- if ($outfile ne '-') {
- close($fh) or die sprintf(__("%s: error on closing %s: %s\n"),
- $real_command_name, $outfile, $!);
- }
- if ($base_level > 0) {
- if (!$new->content_seen) {
- # this should only happen for input coming from pipe or the like
- warn sprintf(__("%s: removing %s as input file %s has no content\n"),
- $real_command_name, $outfile, $file);
- unlink ($outfile);
- pop @included;
- # if we didn't gather the short title, try now, and rename out file if found
- } elsif (!defined($name)) {
- my $short_title = $new->texinfo_short_title;
- if (defined($short_title) and $short_title =~ /\S/) {
- push @manuals, $short_title;
- pop @included;
- my $new_outfile
- = Pod::Simple::Texinfo::_pod_title_to_file_name($short_title);
- $new_outfile .= '.texi';
- $new_outfile = File::Spec->catfile($subdir, $new_outfile)
- if (defined($subdir));
- if ($new_outfile ne $outfile) {
- unless (rename ($outfile, $new_outfile)) {
- die sprintf(__("%s: rename %s failed: %s\n"),
- $real_command_name, $outfile, $!);
- }
- }
- push @included, [$short_title, $new_outfile, $file];
- }
- }
- }
- $file_nr++;
- }
- if ($base_level > 0) {
- my $fh;
- if ($output ne '-') {
- open (OUT, ">$output") or die sprintf(__("%s: could not open %s for writing: %s\n"),
- $real_command_name, $output, $!);
- $fh = *OUT;
- } else {
- $fh = *STDOUT;
- }
- # FIXME should use =encoding
- binmode($fh, ':encoding(utf8)');
- my $outfile_name = $output;
- $outfile_name = $STDOUT_DOCU_NAME if ($outfile_name eq '-');
- $outfile_name =~ s/\.te?x(i|info)?$//;
- $outfile_name .= '.info';
- if (! defined ($preamble)) {
- $preamble = '\input texinfo
- @setfilename ' . Pod::Simple::Texinfo::_protect_text($outfile_name) . "
- \@documentencoding utf-8
- \@settitle $top
- \@contents
- \@ifnottex
- \@node Top
- \@top $top
- \@end ifnottex\n\n";
- }
-
- print $fh $preamble;
- if ($section_nodes) {
- #print STDERR "\@node Top\n\@top top\n".$full_manual;
- my $menu = _do_top_node_menu("\@node Top\n\@top top\n".$full_manual);
- print $fh $menu."\n";
- }
- foreach my $include (@included) {
- my $file = $include->[1];
- print $fh "\@include ".Pod::Simple::Texinfo::_protect_text($file)."\n";
- }
- print $fh "\n\@bye\n";
-
- if ($output ne '-') {
- close($fh) or die sprintf(__("%s: error on closing %s: %s\n"),
- $real_command_name, $output, $!);
- }
- }
- if (defined($output) and $output eq '-') {
- close(STDOUT) or die sprintf(__("%s: error on closing stdout: %s\n"),
- $real_command_name, $!);
- }
- 1;
- __END__
- =head1 NAME
- pod2texi - convert Pod to Texinfo
- =head1 SYNOPSIS
- pod2texi [OPTION]... POD...
- =head1 DESCRIPTION
- Translate Pod file(s) to Texinfo. There are two basic modes of
- operation. First, by default, each pod is translated to a standalone
- Texinfo manual.
- Second, if C<--base-level> is set higher than 0, each pod is translated
- to a file suitable for C<@include>, and one more file with all the
- C<@include>s is generated, intended to be C<@include>d in turn within a
- hand-written top-level file.
- =head1 OPTIONS
- =over
- =item B<--appendix-sections>
- Use appendix sectioning commands (C<@appendix>, ...) instead of the
- default numbered sectioning Texinfo @-commands (C<@chapter>,
- C<@section>, ...).
- =item B<--base-level>=I<NUM|NAME>
- Sets the level of the C<head1> commands. It may be an integer or a
- Texinfo sectioning command (without the C<@>): 1 corresponds to the
- C<@chapter>/C<@unnumbered> level, 2 to the C<@section> level, and so on.
- The default is 0, meaning that C<head1> commands are still output as
- chapters, but the output is arranged as a standalone manual.
- If the level is not 0, the pod file is rendered as a fragment of a
- Texinfo manual suitable for C<@include>. In this case, each pod file
- has an additional sectioning command covering the entire file, one level
- above the C<--base-level> value. Therefore, to make each pod file a
- chapter in a large manual, you should use C<section> as the base level.
- For an example of making Texinfo out of the Perl documentation itself,
- see C<contrib/perldoc-all> in the Texinfo source distribution, with
- output available at L<http://www.gnu.org/software/perl/manual>.
- =item B<--debug>=I<NUM>
- Set debugging level to I<NUM>.
- =item B<--help>
- Display help and exit.
- =item B<--output>=I<NAME>
- Name for the first manual, or the main manual if there is a main manual.
- Default is to write to standard output.
- =item B<--no-section-nodes>
- Use anchors for sections instead of nodes.
- =item B<--no-fill-section-gaps>
- Do not fill sectioning gaps with empty C<@unnumbered> files.
- Ordinarily, it's good to keep the sectioning hierarchy intact.
- =item B<--preamble>=I<STR>
- Insert I<STR> as top boilerplate before includes. The default is a
- minimal beginning for a Texinfo document, and sets C<@documentencoding>
- to C<utf-8> (because the output is written that way).
- =item B<--subdir>=I<NAME>
- If there is a main manual with include files (each corresponding to
- an input pod file), then those include files are put in directory I<NAME>.
- =item B<--unnumbered-sections>
- Use unnumbered sectioning commands (C<@unnumbered>, ...) instead of the
- default numbered sectioning Texinfo @-commands (C<@chapter>,
- C<@section>, ...).
- =item B<--top>=I<TOP>
- Name of the C<@top> element for the main manual. May contain Texinfo code.
- =item B<--version>
- Display version information and exit.
- =back
- =head1 SEE ALSO
- L<Pod::Simple::Texinfo>. L<perlpod>. The Texinfo manual.
- Texinfo home page: L<http://www.gnu.org/software/texinfo/>
- =head1 COPYRIGHT
- Copyright 2016 Free Software Foundation, Inc.
- This program is free software; you can redistribute it and/or modify
- it under the terms of the GNU General Public License as published by
- the Free Software Foundation; either version 3 of the License,
- or (at your option) any later version.
- There is NO WARRANTY, to the extent permitted by law.
- =head1 AUTHOR
- Patrice Dumas E<lt>bug-texinfo@gnu.orgE<gt>.
- =cut
|