This commit is contained in:
2019-02-06 00:49:12 +03:00
commit 8dbb1bb605
4796 changed files with 506072 additions and 0 deletions

243
wcmtools/bin/apidoc.pl Executable file
View File

@@ -0,0 +1,243 @@
#!/usr/bin/perl
# This script parses LJ function info from all the library files
# that make up the site. See cgi-bin/ljlib.pl for an example
# of the necessary syntax.
use strict;
use Getopt::Long;
use Data::Dumper;
my $opt_warn = 0;
my $opt_file;
my $opt_stubs = 0; # generate stubs of undoced funcs
my $opt_class = 0; # group by class
my ($opt_include, $opt_exclude); # which packages to inc/excl
my @do_dirs;
my $basedir;
my $opt_conf;
die unless GetOptions(
'warn' => \$opt_warn,
'file=s' => \$opt_file,
'stubs' => \$opt_stubs,
'class' => \$opt_class,
'include=s' => \$opt_include,
'exclude=s' => \$opt_exclude,
'conf=s' => \$opt_conf,
);
die "Unknown arguments.\n" if @ARGV;
die "Can't exclude and include at same time!\n" if $opt_include && $opt_exclude;
my (@classes, %classname, %common_args);
if ($opt_conf) {
open (C, $opt_conf) or die "Can't open conf file: $opt_conf\n";
while (<C>)
{
chomp;
if (/^basedir\s+(\S+)$/) {
$basedir = $1;
$basedir =~ s/\$(\w+)/$ENV{$1} or die "Undefined ENV: $1"/eg;
} elsif (/^dodir\s+(\S+)$/) {
push @do_dirs, $1;
} elsif (/^class\s+(\w+)\s+(.+)/) {
push @classes, $1;
$classname{$1} = $2;
} elsif (/^arg\s+(\S+)\s+(.+)/) {
$common_args{$1} = $2;
} elsif (/\S/) {
die "Unknown line in conf file:\n$_\n";
}
}
close C;
}
my %funcs;
if ($opt_file) {
check_file($opt_file);
} else {
unless ($basedir) {
die "No base directory specified.\n";
}
chdir $basedir or die "Can't cd to base: $basedir\n";
foreach (@do_dirs) {
find($_);
}
}
exit if $opt_warn;
if ($opt_class)
{
my %by_class;
foreach my $n (sort keys %funcs) {
my $f = $funcs{$n};
push @{$by_class{$f->{'class'}}}, $f;
}
my $ret = [];
foreach my $cn (@classes) {
push @$ret, [ $classname{$cn}, $by_class{$cn} ];
}
print Dumper($ret);
exit;
}
print Dumper(\%funcs);
exit;
sub find
{
my @dirs = @_;
while (@dirs)
{
my $dir = shift @dirs;
opendir (D, $dir) or die "Can't open dir: $dir\n";
my @files = sort { $a cmp $b } readdir(D);
close D;
foreach my $f (@files) {
next if ($f eq "." || $f eq "..");
my $full = "$dir/$f";
if (-d $full) { find($full); }
elsif (-f $full) { check_file($full); }
}
}
}
sub check_file
{
$_ = shift;
return unless (-f);
return if (/\.(gif|jpg|png|class|jar|zip|exe|orig|rej)$/);
return if (/~$/);
my $curpackage = "";
my $file = $_;
my $infunc = 0;
my $f; # the current function info we're loading
my $prefix;
my $curkey;
my $contlen;
open (F, $file) or die "Can't open file: $file\n";
while (my $l = <F>)
{
if ($l =~ /^package\s*(.+);/) {
$curpackage = $1;
}
if ($opt_warn && $curpackage && $l =~ /^sub\s+([a-zA-Z0-9]\S+)/) {
my $s = $1;
my $total = $curpackage . "::" . $s;
unless ($funcs{$total}) {
print STDERR "Undocumented: $total\n";
if ($opt_stubs) {
print "# <LJFUNC>\n";
print "# name: $total\n";
print "# class: \n";
print "# des: \n";
print "# info: \n";
print "# args: \n";
print "# des-: \n";
print "# returns: \n";
print "# </LJFUNC>\n";
}
}
}
print $l if $opt_stubs;
if (! $infunc) {
if ($l =~ /<LJFUNC>/) {
$infunc = 1;
$f = {};
}
next;
}
if ($l =~ /<\/LJFUNC>/) {
$infunc = 0;
$prefix = "";
$curkey = "";
$contlen = 0;
my $include = 0;
if ($opt_exclude) {
$include = 1;
$include = 0 if $f->{'name'} =~ /^$opt_exclude/;
} elsif ($opt_include) {
$include = 1 if $f->{'name'} =~ /^$opt_include/;
} elsif (! $opt_include && ! $opt_exclude) {
$include = 1;
}
if ($f->{'name'} && $include) {
$f->{'source'} = $file;
$f->{'class'} ||= "general";
unless ($classname{$f->{'class'}}) {
print STDERR "Unknown class: $f->{'class'} ($f->{'name'})\n";
}
$funcs{$f->{'name'}} = $f;
treeify($f);
}
next;
}
# continuing a line from line before... must have
# same indenting.
if ($prefix && $contlen) {
my $cont = $prefix . " "x$contlen;
if ($l =~ /^\Q$cont\E(.+)/) {
my $v = $1;
$v =~ s/^\s+//;
$v =~ s/\s+$//;
$f->{$curkey} .= " " . $v;
next;
}
}
if ($l =~ /^(\W*)([\w\-]+)(:\s*)(.+)/) {
$prefix = $1;
my $k = $2;
my $v = $4;
$v =~ s/^\s+//;
$v =~ s/\s+$//;
$f->{$k} = $v;
$curkey = $k;
$contlen = length($2) + length($3);
}
}
close (F);
}
sub treeify
{
my $f = shift;
my $args = $f->{'args'};
$f->{'args'} = [];
$args =~ s/\s+//g;
foreach my $arg (split(/\,/, $args))
{
my $opt = 0;
if ($arg =~ s/\?$//) { $opt = 1; }
my $list = 0;
if ($arg =~ s/\*$//) { $list = 1; }
my $a = { 'name' => $arg };
if ($opt) { $a->{'optional'} = 1; }
if ($list) { $a->{'list'} = 1; }
$a->{'des'} = $f->{"des-$arg"} || $common_args{$arg};
delete $f->{"des-$arg"};
unless ($a->{'des'}) {
if ($opt_warn) {
print "Warning: undescribed argument '$arg' in $a->{'name'}\n";
}
}
push @{$f->{'args'}}, $a;
}
}

268
wcmtools/bin/multicvs.pl Executable file
View File

@@ -0,0 +1,268 @@
#!/usr/bin/perl
#
use strict;
use Getopt::Long;
$| = 1;
my $help = 0;
my $sync = 0;
my $diff = 0;
my $cvsonly = 0;
my $liveonly = 0;
my $init = 0;
my $conf;
my $opt_update;
my $opt_justfiles;
my $opt_ignore_space;
my $these_flag;
exit 1 unless GetOptions('conf=s' => \$conf,
'help' => \$help,
'sync' => \$sync,
'diff' => \$diff,
'cvsonly|c' => \$cvsonly,
'liveonly' => \$liveonly,
'init' => \$init,
'update' => \$opt_update,
'justfiles|1' => \$opt_justfiles,
'no-space-changes|b|w' => \$opt_ignore_space,
'these|t' => \$these_flag,
);
if ($help or not defined $conf) {
die "Usage: multicvs.pl --conf=/path/to/multicvs.conf [opts] [files]\n" .
" --help Get this help\n" .
" --sync Put files where they need to go.\n" .
" All files, unless you specify which ones.\n".
" --diff Show diffs of changed files.\n".
" --cvsonly Don't consider files changed in live dirs.\n".
" --liveonly Don't consider files changed in the CVS dirs.\n".
" --init Copy all files from cvs to main, unconditionally.\n" .
" --update Updates files in the CVS dirs from the cvs repositories.\n".
" --justfiles -1 Only output files, not the old -> new arrow. (good for xargs)\n".
" --no-space-changes -b Do not display whitespace differences.\n".
" --these -t Refuse to --sync if no files are specified.\n";
}
if ($init) {
$sync = 1;
die "Can't set --liveonly or --cvsonly with --init\n"
if $cvsonly or $liveonly;
$diff = 0;
}
unless (-e $conf) {
die "Specified conf file doesn't exist: $conf\n";
}
my ($DIR_LIVE, $DIR_CVS);
my @paths;
my $read_conf = sub
{
my $file = shift;
my $main = shift;
open (C, $file) or die "Error opening conf file.\n";
while (<C>)
{
s/\#.*//;
next unless /\S/;
s/^\s+//;
s/\s+$//;
s/\$(\w+)/$ENV{$1} or die "Environment variable \$$1 not set.\n"/ge;
if (/(\w+)\s*=\s*(.+)/) {
my ($k, $v) = ($1, $2);
unless ($main) {
die "Included config files can't set variables such as $k.\n";
}
if ($k eq "LIVEDIR") { $DIR_LIVE = $v; }
elsif ($k eq "CVSDIR") { $DIR_CVS = $v; }
else { die "Unknown option $k = $v\n"; }
next;
}
if (/(\S+)\s+(.+)/) {
my ($from, $to) = ($1, $2);
my $maybe = 0;
if ($from =~ s/\?$//) { $maybe = 1; }
push @paths, {
'from' => $from,
'to' => $to,
'maybe' => $maybe,
};
} else {
die "Bogus line: $_\n";
}
}
close C;
};
$read_conf->($conf, 1);
if ($conf =~ /^(.+)(multicvs\.conf)$/) {
my $localconf = "$1multicvs-local.conf";
$read_conf->($localconf) if -e $localconf;
}
my %cvspath; # live path -> cvs path
my %have_updated;;
foreach my $p (@paths)
{
unless (-e "$DIR_CVS/$p->{'from'}") {
warn "WARNING: $p->{'from'} doesn't exist under $DIR_CVS\n"
unless $p->{'maybe'};
next;
}
if ($opt_update) {
my $root = $p->{'from'};
$root =~ s!/.*!!;
my $dir = "$DIR_CVS/$root";
if (-d $dir && ! $have_updated{$dir}) {
chdir $dir or die "Can't cd to $dir\n";
print "Updating CVS dir '$root' ...\n";
system("cvs", "update", "-dP");
$have_updated{$dir} = 1;
}
}
if (-f "$DIR_CVS/$p->{'from'}") {
$cvspath{$p->{'to'}} = $p->{'from'};
next;
}
$p->{'to'} =~ s!/$!!;
my $to_prefix = "$p->{'to'}/";
$to_prefix =~ s!^\./!!;
my @dirs = ($p->{'from'});
while (@dirs)
{
my $dir = shift @dirs;
my $fulldir = "$DIR_CVS/$dir";
opendir (MD, $fulldir) or die "Can't open $fulldir.";
while (my $file = readdir(MD)) {
next if ($file =~ /~$/); # ignore emacs files
next if ($file =~ /^\.\#/); # ignore CVS archived versions
next if ($file =~ /\bCVS\b/);
next if $file eq "." or $file eq "..";
if (-d "$fulldir/$file") {
unshift @dirs, "$dir/$file";
} elsif (-f "$fulldir/$file") {
my $to = "$dir/$file";
$to =~ s!^$p->{'from'}/!!;
$cvspath{"$to_prefix$to"} = "$dir/$file";
}
}
close MD;
}
}
# If the user has specified that there must be arguments, require @ARGV to
# contain soemthing.
die "These what?\n\nWith --these specified, you must provide at least one file to sync.\n"
if $these_flag && $sync && !@ARGV;
my @files = scalar(@ARGV) ? @ARGV : sort keys %cvspath;
foreach my $relfile (@files)
{
my $status;
next unless exists $cvspath{$relfile};
my $root = $cvspath{$relfile};
$root =~ s!/.*!!;
my ($from, $to); # if set, do action (diff and/or sync)
my $lfile = "$DIR_LIVE/$relfile";
my $cfile = "$DIR_CVS/$cvspath{$relfile}";
if ($init) {
$status = "main <- $root";
($from, $to) = ($cfile, $lfile);
} else {
my $ltime = mtime($lfile);
my $ctime = mtime($cfile);
next if $ltime == $ctime;
if ($ltime > $ctime && ! $cvsonly) {
$status = "main -> $root";
($from, $to) = ($lfile, $cfile);
}
if ($ctime > $ltime && ! $liveonly) {
$status = "main <- $root";
($from, $to) = ($cfile, $lfile);
}
}
next unless $status;
my $the_diff;
if ($diff && -e $from && -e $to) {
my $opt;
$opt = '-b' if $opt_ignore_space;
$the_diff = `diff -u $opt $to $from`; # getting from destination to source
if ($the_diff) {
# fix the -p level to be -p0
my $slashes = ($DIR_LIVE =~ tr!/!/!);
$the_diff =~ s/((^|\n)[\-\+]{3,3} )\/([^\/]+?\/){$slashes,$slashes}/$1/g;
} else {
# don't touch the files that don't have a diff if we're ignoring spaces
# as there might really be one and we just don't see it
next if $opt_ignore_space;
# no real change (just touched/copied?), so copy
# cvs one on top to fix times up.
copy($from, $to);
next;
}
}
if ($sync) {
make_dirs($relfile);
copy($from, $to);
}
if ($opt_justfiles) {
print "$relfile\n";
} else {
printf "%-25s %s\n", $status, $relfile;
print $the_diff;
}
}
sub mtime
{
my $file = shift;
return (stat($file))[9];
}
my %MADE_DIR;
sub make_dirs
{
my $file = shift;
return 1 unless $file =~ s!/[^/]*$!!;
return 1 if $MADE_DIR{$file};
my @dirs = split(m!/!, $file);
for (my $i=0; $i<scalar(@dirs); $i++) {
my $sd = join("/", @dirs[0..$i]);
my $makedir = "$DIR_LIVE/$sd";
unless (-d $makedir) {
mkdir $makedir, 0755
or die "Couldn't make directory $makedir\n";
}
}
$MADE_DIR{$file} = 1;
}
# was using perl's File::Copy, but I want to preserve the file time.
sub copy
{
my ($src, $dest) = @_;
my $ret = system("cp", "-p", $src, $dest);
return ($ret == 0);
}
__END__