File:  [ELWIX - Embedded LightWeight unIX -] / embedaddon / curl / docs / cmdline-opts / gen.pl
Revision 1.1.1.1 (vendor branch): download - view: text, annotated - select for diffs - revision graph
Wed Jun 3 10:01:15 2020 UTC (4 years, 10 months ago) by misho
Branches: curl, MAIN
CVS tags: v7_70_0p4, HEAD
curl

    1: #!/usr/bin/env perl
    2: #***************************************************************************
    3: #                                  _   _ ____  _
    4: #  Project                     ___| | | |  _ \| |
    5: #                             / __| | | | |_) | |
    6: #                            | (__| |_| |  _ <| |___
    7: #                             \___|\___/|_| \_\_____|
    8: #
    9: # Copyright (C) 1998 - 2020, Daniel Stenberg, <daniel@haxx.se>, et al.
   10: #
   11: # This software is licensed as described in the file COPYING, which
   12: # you should have received as part of this distribution. The terms
   13: # are also available at https://curl.haxx.se/docs/copyright.html.
   14: #
   15: # You may opt to use, copy, modify, merge, publish, distribute and/or sell
   16: # copies of the Software, and permit persons to whom the Software is
   17: # furnished to do so, under the terms of the COPYING file.
   18: #
   19: # This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
   20: # KIND, either express or implied.
   21: #
   22: ###########################################################################
   23: 
   24: =begin comment
   25: 
   26: This script generates the manpage.
   27: 
   28: Example: gen.pl <command> [files] > curl.1
   29: 
   30: Dev notes:
   31: 
   32: We open *input* files in :crlf translation (a no-op on many platforms) in
   33: case we have CRLF line endings in Windows but a perl that defaults to LF.
   34: Unfortunately it seems some perls like msysgit can't handle a global input-only
   35: :crlf so it has to be specified on each file open for text input.
   36: 
   37: =end comment
   38: =cut
   39: 
   40: my %optshort;
   41: my %optlong;
   42: my %helplong;
   43: my %arglong;
   44: my %redirlong;
   45: my %protolong;
   46: 
   47: # get the long name version, return the man page string
   48: sub manpageify {
   49:     my ($k)=@_;
   50:     my $l;
   51:     if($optlong{$k} ne "") {
   52:         # both short + long
   53:         $l = "\\fI-".$optlong{$k}.", --$k\\fP";
   54:     }
   55:     else {
   56:         # only long
   57:         $l = "\\fI--$k\\fP";
   58:     }
   59:     return $l;
   60: }
   61: 
   62: sub printdesc {
   63:     my @desc = @_;
   64:     for my $d (@desc) {
   65:         # skip lines starting with space (examples)
   66:         if($d =~ /^[^ ]/) {
   67:             for my $k (keys %optlong) {
   68:                 my $l = manpageify($k);
   69:                 $d =~ s/--$k([^a-z0-9_-])/$l$1/;
   70:             }
   71:         }
   72:         print $d;
   73:     }
   74: }
   75: 
   76: sub seealso {
   77:     my($standalone, $data)=@_;
   78:     if($standalone) {
   79:         return sprintf
   80:             ".SH \"SEE ALSO\"\n$data\n";
   81:     }
   82:     else {
   83:         return "See also $data. ";
   84:     }
   85: }
   86: 
   87: sub overrides {
   88:     my ($standalone, $data)=@_;
   89:     if($standalone) {
   90:         return ".SH \"OVERRIDES\"\n$data\n";
   91:     }
   92:     else {
   93:         return $data;
   94:     }
   95: }
   96: 
   97: sub protocols {
   98:     my ($standalone, $data)=@_;
   99:     if($standalone) {
  100:         return ".SH \"PROTOCOLS\"\n$data\n";
  101:     }
  102:     else {
  103:         return "($data) ";
  104:     }
  105: }
  106: 
  107: sub added {
  108:     my ($standalone, $data)=@_;
  109:     if($standalone) {
  110:         return ".SH \"ADDED\"\nAdded in curl version $data\n";
  111:     }
  112:     else {
  113:         return "Added in $data. ";
  114:     }
  115: }
  116: 
  117: sub single {
  118:     my ($f, $standalone)=@_;
  119:     open(F, "<:crlf", "$f") ||
  120:         return 1;
  121:     my $short;
  122:     my $long;
  123:     my $tags;
  124:     my $added;
  125:     my $protocols;
  126:     my $arg;
  127:     my $mutexed;
  128:     my $requires;
  129:     my $seealso;
  130:     my $magic; # cmdline special option
  131:     while(<F>) {
  132:         if(/^Short: *(.)/i) {
  133:             $short=$1;
  134:         }
  135:         elsif(/^Long: *(.*)/i) {
  136:             $long=$1;
  137:         }
  138:         elsif(/^Added: *(.*)/i) {
  139:             $added=$1;
  140:         }
  141:         elsif(/^Tags: *(.*)/i) {
  142:             $tags=$1;
  143:         }
  144:         elsif(/^Arg: *(.*)/i) {
  145:             $arg=$1;
  146:         }
  147:         elsif(/^Magic: *(.*)/i) {
  148:             $magic=$1;
  149:         }
  150:         elsif(/^Mutexed: *(.*)/i) {
  151:             $mutexed=$1;
  152:         }
  153:         elsif(/^Protocols: *(.*)/i) {
  154:             $protocols=$1;
  155:         }
  156:         elsif(/^See-also: *(.*)/i) {
  157:             $seealso=$1;
  158:         }
  159:         elsif(/^Requires: *(.*)/i) {
  160:             $requires=$1;
  161:         }
  162:         elsif(/^Help: *(.*)/i) {
  163:             ;
  164:         }
  165:         elsif(/^---/) {
  166:             if(!$long) {
  167:                 print STDERR "WARN: no 'Long:' in $f\n";
  168:             }
  169:             last;
  170:         }
  171:         else {
  172:             chomp;
  173:             print STDERR "WARN: unrecognized line in $f, ignoring:\n:'$_';"
  174:         }
  175:     }
  176:     my @desc;
  177:     while(<F>) {
  178:         push @desc, $_;
  179:     }
  180:     close(F);
  181:     my $opt;
  182:     if(defined($short) && $long) {
  183:         $opt = "-$short, --$long";
  184:     }
  185:     elsif($short && !$long) {
  186:         $opt = "-$short";
  187:     }
  188:     elsif($long && !$short) {
  189:         $opt = "--$long";
  190:     }
  191: 
  192:     if($arg) {
  193:         $opt .= " $arg";
  194:     }
  195: 
  196:     if($standalone) {
  197:         print ".TH curl 1 \"30 Nov 2016\" \"curl 7.52.0\" \"curl manual\"\n";
  198:         print ".SH OPTION\n";
  199:         print "curl $opt\n";
  200:     }
  201:     else {
  202:         print ".IP \"$opt\"\n";
  203:     }
  204:     if($protocols) {
  205:         print protocols($standalone, $protocols);
  206:     }
  207: 
  208:     if($standalone) {
  209:         print ".SH DESCRIPTION\n";
  210:     }
  211: 
  212:     printdesc(@desc);
  213:     undef @desc;
  214: 
  215:     my @foot;
  216:     if($seealso) {
  217:         my @m=split(/ /, $seealso);
  218:         my $mstr;
  219:         for my $k (@m) {
  220:             if(!$helplong{$k}) {
  221:                 print STDERR "WARN: $f see-alsos a non-existing option: $k\n";
  222:             }
  223:             my $l = manpageify($k);
  224:             $mstr .= sprintf "%s$l", $mstr?" and ":"";
  225:         }
  226:         push @foot, seealso($standalone, $mstr);
  227:     }
  228:     if($requires) {
  229:         my $l = manpageify($long);
  230:         push @foot, "$l requires that the underlying libcurl".
  231:             " was built to support $requires. ";
  232:     }
  233:     if($mutexed) {
  234:         my @m=split(/ /, $mutexed);
  235:         my $mstr;
  236:         for my $k (@m) {
  237:             if(!$helplong{$k}) {
  238:                 print STDERR "WARN: $f mutexes a non-existing option: $k\n";
  239:             }
  240:             my $l = manpageify($k);
  241:             $mstr .= sprintf "%s$l", $mstr?" and ":"";
  242:         }
  243:         push @foot, overrides($standalone, "This option overrides $mstr. ");
  244:     }
  245:     if($added) {
  246:         push @foot, added($standalone, $added);
  247:     }
  248:     if($foot[0]) {
  249:         print "\n";
  250:         my $f = join("", @foot);
  251:         $f =~ s/ +\z//; # remove trailing space
  252:         print "$f\n";
  253:     }
  254:     return 0;
  255: }
  256: 
  257: sub getshortlong {
  258:     my ($f)=@_;
  259:     open(F, "<:crlf", "$f");
  260:     my $short;
  261:     my $long;
  262:     my $help;
  263:     my $arg;
  264:     my $protocols;
  265:     while(<F>) {
  266:         if(/^Short: (.)/i) {
  267:             $short=$1;
  268:         }
  269:         elsif(/^Long: (.*)/i) {
  270:             $long=$1;
  271:         }
  272:         elsif(/^Help: (.*)/i) {
  273:             $help=$1;
  274:         }
  275:         elsif(/^Arg: (.*)/i) {
  276:             $arg=$1;
  277:         }
  278:         elsif(/^Protocols: (.*)/i) {
  279:             $protocols=$1;
  280:         }
  281:         elsif(/^---/) {
  282:             last;
  283:         }
  284:     }
  285:     close(F);
  286:     if($short) {
  287:         $optshort{$short}=$long;
  288:     }
  289:     if($long) {
  290:         $optlong{$long}=$short;
  291:         $helplong{$long}=$help;
  292:         $arglong{$long}=$arg;
  293:         $protolong{$long}=$protocols;
  294:     }
  295: }
  296: 
  297: sub indexoptions {
  298:     my (@files) = @_;
  299:     foreach my $f (@files) {
  300:         getshortlong($f);
  301:     }
  302: }
  303: 
  304: sub header {
  305:     my ($f)=@_;
  306:     open(F, "<:crlf", "$f");
  307:     my @d;
  308:     while(<F>) {
  309:         push @d, $_;
  310:     }
  311:     close(F);
  312:     printdesc(@d);
  313: }
  314: 
  315: sub listhelp {
  316:     foreach my $f (sort keys %helplong) {
  317:         my $long = $f;
  318:         my $short = $optlong{$long};
  319:         my $opt;
  320: 
  321:         if(defined($short) && $long) {
  322:             $opt = "-$short, --$long";
  323:         }
  324:         elsif($long && !$short) {
  325:             $opt = "    --$long";
  326:         }
  327: 
  328:         my $arg = $arglong{$long};
  329:         if($arg) {
  330:             $opt .= " $arg";
  331:         }
  332:         my $desc = $helplong{$f};
  333:         $desc =~ s/\"/\\\"/g; # escape double quotes
  334: 
  335:         my $line = sprintf "  {\"%s\",\n   \"%s\"},\n", $opt, $desc;
  336: 
  337:         if(length($opt) + length($desc) > 78) {
  338:             print STDERR "WARN: the --$long line is too long\n";
  339:         }
  340:         print $line;
  341:     }
  342: }
  343: 
  344: sub mainpage {
  345:     my (@files) = @_;
  346:     # show the page header
  347:     header("page-header");
  348: 
  349:     # output docs for all options
  350:     foreach my $f (sort @files) {
  351:         if(single($f, 0)) {
  352:             print STDERR "Can't read $f?\n";
  353:         }
  354:     }
  355: 
  356:     header("page-footer");
  357: }
  358: 
  359: sub showonly {
  360:     my ($f) = @_;
  361:     if(single($f, 1)) {
  362:         print STDERR "$f: failed\n";
  363:     }
  364: }
  365: 
  366: sub showprotocols {
  367:     my %prots;
  368:     foreach my $f (keys %optlong) {
  369:         my @p = split(/ /, $protolong{$f});
  370:         for my $p (@p) {
  371:             $prots{$p}++;
  372:         }
  373:     }
  374:     for(sort keys %prots) {
  375:         printf "$_ (%d options)\n", $prots{$_};
  376:     }
  377: }
  378: 
  379: sub getargs {
  380:     my ($f, @s) = @_;
  381:     if($f eq "mainpage") {
  382:         mainpage(@s);
  383:         return;
  384:     }
  385:     elsif($f eq "listhelp") {
  386:         listhelp();
  387:         return;
  388:     }
  389:     elsif($f eq "single") {
  390:         showonly($s[0]);
  391:         return;
  392:     }
  393:     elsif($f eq "protos") {
  394:         showprotocols();
  395:         return;
  396:     }
  397: 
  398:     print "Usage: gen.pl <mainpage/listhelp/single FILE/protos> [files]\n";
  399: }
  400: 
  401: #------------------------------------------------------------------------
  402: 
  403: my $cmd = shift @ARGV;
  404: my @files = @ARGV; # the rest are the files
  405: 
  406: # learn all existing options
  407: indexoptions(@files);
  408: 
  409: getargs($cmd, @files);

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>