aboutsummaryrefslogtreecommitdiffstats
path: root/Tools/scripts
diff options
context:
space:
mode:
Diffstat (limited to 'Tools/scripts')
-rwxr-xr-xTools/scripts/chkversion.pl173
1 files changed, 138 insertions, 35 deletions
diff --git a/Tools/scripts/chkversion.pl b/Tools/scripts/chkversion.pl
index 63f48e90ad7..917cc3a8586 100755
--- a/Tools/scripts/chkversion.pl
+++ b/Tools/scripts/chkversion.pl
@@ -45,12 +45,20 @@
#
# To use the script as intended, do the following (assuming you want to
# run the script as user `ports'):
+#
# install port sysutils/pkg_install-devel (optional)
# mkdir -p /var/db/chkversion
# touch /var/db/chkversion/VERSIONS
# chown -R ports /var/db/chkversion
-# and enter chkversions.pl into the crontab of ports, or run it by hand
-# if you can spare the time.
+# and enter something like
+#
+# CVSBLAME=yes
+# ALLPORTS=yes
+# RCPT_ORIGIN=you@domain.example
+# RCPT_VERSION=you@domain.example
+# 0 */2 * * * /usr/ports/Tools/scripts/chkversion.pl
+#
+# into `crontab -u ports -e', or run the script by hand if you can spare the time.
#
# If the environment variable CVSBLAME is set and the ports tree is checked
# out by CVS, every entry is listed with a record of the last CVS commit.
@@ -61,10 +69,18 @@ use strict;
use File::Find;
use Cwd 'abs_path';
-my $portsdir = $ENV{PORTSDIR} ? $ENV{PORTSDIR} : '/usr/ports';
-my $versiondir = $ENV{VERSIONDIR} ? $ENV{VERSIONDIR} : '/var/db/chkversion';
-my $cvsblame = $ENV{CVSBLAME} ? 1 : 0;
-my $allports = $ENV{ALLPORTS} ? 1 : 0;
+my $portsdir = $ENV{PORTSDIR} ? $ENV{PORTSDIR} : '/usr/ports';
+my $versiondir = $ENV{VERSIONDIR} ? $ENV{VERSIONDIR} : '/var/db/chkversion';
+my $cvsblame = $ENV{CVSBLAME} ? 1 : 0;
+my $allports = $ENV{ALLPORTS} ? 1 : 0;
+
+my $watchre = $ENV{WATCH_REGEX} ? $ENV{WATCH_REGEX} : '';
+my $returnpath = $ENV{RETURNPATH} ? $ENV{RETURNPATH} : '';
+my $h_from = $ENV{HEADER_FROM} ? $ENV{HEADER_FROM} : "$ENV{USER}\@$ENV{HOST}";
+my $h_replyto = $ENV{HEADER_REPLYTO} ? $ENV{HEADER_REPLYTO} : $h_from;
+my $rcpt_watch = $ENV{RCPT_WATCH} ? $ENV{RCPT_WATCH} : '';
+my $rcpt_orig = $ENV{RCPT_ORIGIN} ? $ENV{RCPT_ORIGIN} : '';
+my $rcpt_vers = $ENV{RCPT_VERSION} ? $ENV{RCPT_VERSION} : '';
my $make = '/usr/bin/make';
my $cvs = '/usr/bin/cvs';
@@ -72,6 +88,9 @@ my $pkg_version =
$ENV{PKG_VERSION} && -x $ENV{PKG_VERSION} ? $ENV{PKG_VERSION}
: -x '/usr/local/sbin/pkg_version' ? '/usr/local/sbin/pkg_version'
: '/usr/sbin/pkg_version';
+my $sendmail = '/usr/sbin/sendmail';
+
+my $watch_re = join "|", split " ", $watchre;
-d $portsdir or die "Can't find ports tree at $portsdir.\n";
$portsdir = abs_path($portsdir);
@@ -104,6 +123,7 @@ $ENV{WITH_OPENSSL_BASE} = 'yes';
my %pkgname;
my %pkgorigin;
+my %maintainer;
sub wanted {
return
@@ -117,6 +137,7 @@ sub wanted {
$File::Find::prune = 1;
}
elsif ($File::Find::name =~ m"^$portsdir/([^/]+/[^/]+)$"os) {
+ $File::Find::prune = 1;
my @makevar = readfrom $File::Find::name,
$make, '-VPKGORIGIN', '-VPKGNAME';
@@ -124,8 +145,8 @@ sub wanted {
if $makevar[0] && $1 ne $makevar[0];
$pkgname{$1} = $makevar[1]
if $makevar[1];
-
- $File::Find::prune = 1;
+ $maintainer{$_} = $makevar[2]
+ if $makevar[2];
}
}
@@ -142,16 +163,20 @@ else {
foreach (map "$category/$_", @ports) {
-f "$portsdir/$_/Makefile" || next;
my @makevar = readfrom "$portsdir/$_",
- $make, '-VPKGORIGIN', '-VPKGNAME';
+ $make, '-VPKGORIGIN', '-VPKGNAME', '-VMAINTAINER';
+
$pkgorigin{$_} = $makevar[0]
if $makevar[0] && $_ ne $makevar[0];
$pkgname{$_} = $makevar[1]
if $makevar[1];
+ $maintainer{$_} = $makevar[2]
+ if $makevar[2];
}
}
}
my %backwards;
+my %watched;
if ($useindex) {
my $indexname = readfrom $portsdir, $make, '-VINDEXFILE';
@@ -180,6 +205,10 @@ while (<VERSIONS>) {
my $result = $newversion eq $oldversion ? '=' : readfrom '',
$pkg_version, '-t', $newversion, $oldversion;
+
+ $watched{$origin} = "$version -> $pkgname{$origin}"
+ if ($result ne '=' && $watch_re && $pkgname{$origin} =~ /^(?:$watch_re)$/o);
+
if ($result eq '<') {
$backwards{$origin} = "$pkgname{$origin} < $version";
$pkgname{$origin} = $version;
@@ -195,43 +224,117 @@ if (!$useindex) {
system 'mv', '-f', $versionfile, "$versionfile.bak";
open VERSIONS, ">$versionfile";
- print VERSIONS join("\n", map("$_\t$pkgname{$_}", sort keys %pkgname)),
- "\n";
+ foreach (sort keys %pkgname) {
+ print VERSIONS "$_\t$pkgname{$_}\n";
+ }
close VERSIONS;
}
sub blame {
- my ($msg, $ports) = @_;
+ my ($fh, $ports) = @_;
if (%{$ports}) {
- print "$msg\n";
- foreach (sort keys %{$ports}) {
- print "- $_: $ports->{$_}\n";
- if ($cvsblame && -d "$portsdir/$_/CVS") {
- my @cvslog = readfrom "$portsdir/$_",
+ foreach my $origin (sort keys %{$ports}) {
+ my $maint = $maintainer{$origin} ? " <$maintainer{$origin}>" : '';
+ print $fh "- *$origin*$maint: $ports->{$origin}\n";
+ if ($cvsblame && -d "$portsdir/$origin/CVS") {
+ my @cvslog = readfrom "$portsdir/$origin",
$cvs, '-R', 'log', '-N', '-r.', 'Makefile';
- print " ", join("\n " , grep(/^-/ .. /^=/, @cvslog)), "\n\n";
+ foreach (@cvslog) {
+ my $in_log = /^-/ ... /^=/;
+ print $fh " | $_\n"
+ if ($in_log && $in_log != 1 && $in_log !~ /E0$/);
+ }
+ print $fh "\n";
+ }
+ }
+ print $fh "\n";
+ }
+}
+
+sub template {
+ my ($from, $rcpt, $replyto) = @_;
+
+ my $header = '';
+ while (<main::DATA>) {
+ last if /^\.\n?$/;
+ $_ =~ s/%%FROM%%/$from/og;
+ $_ =~ s/%%RCPT%%/$rcpt/og;
+ $_ =~ s/%%REPLYTO%%/$replyto/og;
+ $header .= $_;
+ }
+ return $header;
+}
+
+sub mail {
+ my ($template, $rcpt, $ports) = @_;
+
+ if (%{$ports}) {
+ if ($rcpt) {
+ if (!open MAIL, '|-') {
+ exec $sendmail, '-oi', '-t', '-f', $returnpath;
+ die;
}
+ print MAIL $template;
+ blame *MAIL, $ports;
+ close MAIL;
+ } else {
+ $template =~ s/^.*?\n\n//os;
+ print $template;
+ blame *STDOUT, $ports;
}
- print "\n";
}
}
-blame
- "** The following ports have a wrong PKGORIGIN **\n\n"
- . " PKGORIGIN connects packaged or installed ports to the directory they\n"
- . " originated from. This is essential for tools like pkg_version or\n"
- . " portupgrade to work correctly. Wrong PKGORIGINs are often caused by a\n"
- . " wrong order of CATEGORIES after a repocopy.\n",
- \%pkgorigin;
-
-blame
- "** The following ports have a version number that sorts before a previous one **\n\n"
- . " For many package tools to work correctly, it is of utmost importance that\n"
- . " version numbers of a port form a monotonic increasing sequence over time.\n"
- . " Refer to the FreeBSD Porter's Handbook, 'Package Naming Conventions' for\n"
- . " more information. Tools that won't work include pkg_version, portupgrade\n"
- . " and portaudit. A common problem is an accidental deletion of PORTEPOCH.\n",
- \%backwards;
+my $tmpl;
+
+$tmpl = template $h_from, $rcpt_orig, $h_replyto;
+mail $tmpl, $rcpt_orig, \%pkgorigin;
+
+$tmpl = template $h_from, $rcpt_vers, $h_replyto;
+mail $tmpl, $rcpt_vers, \%backwards;
+
+$tmpl = template $h_from, $rcpt_watch, $h_replyto;
+mail $tmpl, $rcpt_watch, \%watched;
exit((%pkgorigin || %backwards) ? 1 : 0);
+
+__END__
+From: %%FROM%%
+To: %%RCPT%%
+Reply-To: %%REPLYTO%%
+Subject: Ports with a wrong PKGORIGIN
+
+** The following ports have a wrong PKGORIGIN **
+
+ PKGORIGIN connects packaged or installed ports to the directory they
+ originated from. This is essential for tools like pkg_version or
+ portupgrade to work correctly. Wrong PKGORIGINs are often caused by a
+ wrong order of CATEGORIES after a repocopy.
+
+.
+From: %%FROM%%
+To: %%RCPT%%
+Reply-To: %%REPLYTO%%
+Subject: Ports with version numbers going backwards
+
+** The following ports have a version number that sorts before a previous one **
+
+ For many package tools to work correctly, it is of utmost importance that
+ version numbers of a port form a monotonic increasing sequence over time.
+ Refer to the FreeBSD Porter's Handbook, 'Package Naming Conventions' for
+ more information. Tools that won't work include pkg_version, portupgrade
+ and portaudit. A common problem is an accidental deletion of PORTEPOCH.
+
+.
+From: %%FROM%%
+To: %%RCPT%%
+Reply-To: %%REPLYTO%%
+Subject: Version changes in watched ports
+
+** The following ports have changed version numbers **
+
+ You have requested to be notified of version changes in the following
+ ports:
+
+.