summaryrefslogtreecommitdiff
path: root/pkgtools/pkglint
diff options
context:
space:
mode:
authorwiz <wiz@pkgsrc.org>2004-08-11 11:53:25 +0000
committerwiz <wiz@pkgsrc.org>2004-08-11 11:53:25 +0000
commit3276480074db04abb3f34ec145320a3a9c17a270 (patch)
treea5327a134cc5d2dd7153812f20433814f0a561d4 /pkgtools/pkglint
parent27e69cf5c4e1c2ca992bc25afcb6319db8140a73 (diff)
downloadpkgsrc-3276480074db04abb3f34ec145320a3a9c17a270.tar.gz
Another refactoring patch from Roland Illig:
more oop; move line functions into own module.
Diffstat (limited to 'pkgtools/pkglint')
-rw-r--r--pkgtools/pkglint/files/pkglint.pl222
1 files changed, 136 insertions, 86 deletions
diff --git a/pkgtools/pkglint/files/pkglint.pl b/pkgtools/pkglint/files/pkglint.pl
index 666364fb55e..2bd548d456b 100644
--- a/pkgtools/pkglint/files/pkglint.pl
+++ b/pkgtools/pkglint/files/pkglint.pl
@@ -11,7 +11,7 @@
# Freely redistributable. Absolutely no warranty.
#
# From Id: portlint.pl,v 1.64 1998/02/28 02:34:05 itojun Exp
-# $NetBSD: pkglint.pl,v 1.116 2004/08/10 10:07:20 wiz Exp $
+# $NetBSD: pkglint.pl,v 1.117 2004/08/11 11:53:25 wiz Exp $
#
# This version contains lots of changes necessary for NetBSD packages
# done by Hubert Feyrer <hubertf@netbsd.org>,
@@ -137,6 +137,73 @@ sub is_verbose()
}
#== End of PkgLint::Logging ===============================================
+package PkgLint::FileUtils;
+#==========================================================================
+# This package provides some file handling subroutines. The subroutine
+# load_file reads a file into memory as an array of lines. A line is a
+# record that contains the fields C<file>, C<lineno> and C<text>.
+#==========================================================================
+
+package PkgLint::FileUtils::Line;
+ sub new($$$$) {
+ my ($class, $file, $lineno, $text) = @_;
+ my ($self) = ({});
+ bless($self, $class);
+ $self->_init($file, $lineno, $text);
+ return $self;
+ }
+ sub _init($$$$) {
+ my ($self, $file, $lineno, $text) = @_;
+ $self->{"file"} = $file;
+ $self->{"lineno"} = $lineno;
+ $self->{"text"} = $text;
+ }
+ sub file($) {
+ return shift(@_)->{"file"};
+ }
+ sub lineno($) {
+ return shift(@_)->{"lineno"};
+ }
+ sub text($) {
+ return shift(@_)->{"text"};
+ }
+ sub toString($) {
+ my ($self) = @_;
+ return sprintf("%s:%d: %s", $self->file, $self->line, $self->text);
+ }
+# end of PkgLint::FileUtils::Line
+
+package PkgLint::FileUtils;
+BEGIN {
+ use Exporter;
+ use vars qw(@ISA @EXPORT_OK);
+ @ISA = qw(Exporter);
+ @EXPORT_OK = qw(load_file);
+}
+
+sub load_file($) {
+ my ($fname) = @_;
+ my ($result, $line, $lineno);
+
+ $result = [];
+ open(F, "< $fname") or return undef;
+ $lineno = 0;
+ while (defined($line = <F>)) {
+ $lineno++;
+ $line =~ s/\r*\n*\z//;
+ push(@$result, PkgLint::FileUtils::Line->new($fname, $lineno, $line));
+ }
+ close(F) or return undef;
+ return $result;
+}
+
+#== End of PkgLint::FileUtils =============================================
+
+package main;
+#==========================================================================
+# This is the main package of pkglint. Currently it contains a lot of
+# functionality, but that will be moved into separate packages.
+#==========================================================================
use strict;
use warnings;
@@ -152,6 +219,9 @@ BEGIN {
log_error log_warning log_info
print_summary_and_exit
);
+ import PkgLint::FileUtils qw(
+ load_file
+ );
}
# Start of configuration area
@@ -422,31 +492,11 @@ EOF
# Subroutines common to all checking routines
#
-# Loads a text file completely into memory. Returns undef on error
-# or a reference to an array of lines. A line itself is an array of
-# three components: the originating file, the line number and the
-# contents of the line.
-sub load_file($) {
- my ($fname) = @_;
- my ($result, $line, $lineno);
-
- $result = [];
- open(F, "< $fname") or return undef;
- $lineno = 0;
- while (defined($line = <F>)) {
- $lineno++;
- $line =~ s/\r*\n*\z//;
- push(@$result, [$fname, $lineno, $line]);
- }
- close(F) or return undef;
- return $result;
-}
-
sub checkline_length($$) {
my ($line, $maxlength) = @_;
- if (length($line->[2]) > $maxlength) {
- log_warning($line->[0], $line->[1], "Line too long (should be no more than $maxlength characters).");
+ if (length($line->text) > $maxlength) {
+ log_warning($line->file, $line->lineno, "Line too long (should be no more than $maxlength characters).");
}
return true;
}
@@ -455,10 +505,10 @@ sub checkline_valid_characters($$) {
my ($line, $re_validchars) = @_;
my ($rest);
- ($rest = $line->[2]) =~ s/$re_validchars//g;
+ ($rest = $line->text) =~ s/$re_validchars//g;
if ($rest ne "") {
my @chars = map { $_ = sprintf("0x%02x", ord($_)); } split(//, $rest);
- log_warning($line->[0], $line->[1],
+ log_warning($line->file, $line->lineno,
sprintf("Line contains invalid characters (%s).", join(", ", @chars)));
}
return true;
@@ -467,7 +517,7 @@ sub checkline_valid_characters($$) {
sub checkline_trailing_whitespace($) {
my ($line) = @_;
if ($line =~ /\s+$/) {
- log_warning($line->[0], $line->[1], "Trailing white space.");
+ log_warning($line->file, $line->lineno, "Trailing white space.");
}
return true;
}
@@ -516,16 +566,16 @@ sub checkfile_distinfo($) {
return false;
}
- if ($distinfo->[0]->[2] !~ /^$regex_rcsidstr$/) {
+ if ($distinfo->[0]->text !~ /^$regex_rcsidstr$/) {
log_error($fname, 1, "\$$conf_rcsidstr\$ (and nothing more) expected.");
}
foreach my $line (@$distinfo[1 .. scalar(@$distinfo)-1]) {
- next unless $line->[2] =~ /^(MD5|SHA1|RMD160) \(([^)]+)\) = (.*)$/;
+ next unless $line->text =~ /^(MD5|SHA1|RMD160) \(([^)]+)\) = (.*)$/;
my ($alg, $patch, $sum) = ($1, $2, $3);
if ($patch =~ /~$/) {
- log_warning($line->[0], $line->[1], "possible backup file \"$patch\"?");
+ log_warning($line->file, $line->lineno, "possible backup file \"$patch\"?");
}
if ($patch =~ /^patch-[A-Za-z0-9_]+$/) {
@@ -533,10 +583,10 @@ sub checkfile_distinfo($) {
my $chksum = `sed -e '/\$NetBSD.*/d' $opt_packagedir/$patchdir/$patch | digest $alg`;
$chksum =~ s/\r*\n*\z//;
if ($sum ne $chksum) {
- log_error($line->[0], $line->[1], "checksum of $patch differs. Rerun '$conf_make makepatchsum'.");
+ log_error($line->file, $line->lineno, "checksum of $patch differs. Rerun '$conf_make makepatchsum'.");
}
} else {
- log_error($line->[0], $line->[1], "$patch does not exist.");
+ log_error($line->file, $line->lineno, "$patch does not exist.");
}
$in_distinfo{$patch} = true;
}
@@ -566,19 +616,19 @@ sub checkfile_MESSAGE($) {
log_warning($fname, NO_LINE_NUMBER, "file too short.");
return false;
}
- if ($message->[0]->[2] ne "=" x 75) {
- log_warning($message->[0]->[0], $message->[0]->[1], "expected a line of exactly 75 \"=\" characters.");
+ if ($message->[0]->text ne "=" x 75) {
+ log_warning($message->[0]->file, $message->[0]->lineno, "expected a line of exactly 75 \"=\" characters.");
}
- if ($message->[1]->[2] !~ /^$regex_rcsidstr$/) {
- log_error($message->[1]->[0], $message->[1]->[1], "expected the RCS Id tag.");
+ if ($message->[1]->text !~ /^$regex_rcsidstr$/) {
+ log_error($message->[1]->file, $message->[1]->lineno, "expected the RCS Id tag.");
}
foreach my $line (@$message[2 .. scalar(@$message) - 2]) {
checkline_length($line, 80);
checkline_trailing_whitespace($line);
checkline_valid_characters($line, $regex_validchars);
}
- if ($message->[-1]->[2] ne "=" x 75) {
- log_warning($message->[-1]->[0], $message->[-1]->[1], "expected a line of exactly 75 \"=\" characters.");
+ if ($message->[-1]->text ne "=" x 75) {
+ log_warning($message->[-1]->file, $message->[-1]->lineno, "expected a line of exactly 75 \"=\" characters.");
}
return true;
}
@@ -599,20 +649,20 @@ sub checkfile_PLIST($) {
foreach my $line (@$plist) {
checkline_trailing_whitespace($line);
- if ($line->[2] =~ /<\$ARCH>/) {
- log_warning($line->[0], $line->[1], "use of <\$ARCH> is deprecated, use \${MACHINE_ARCH} instead.");
+ if ($line->text =~ /<\$ARCH>/) {
+ log_warning($line->file, $line->text, "use of <\$ARCH> is deprecated, use \${MACHINE_ARCH} instead.");
}
- if ($line->[2] =~ /^\@([a-z]+)\s+(.*)/) {
+ if ($line->text =~ /^\@([a-z]+)\s+(.*)/) {
my ($cmd, $arg) = ($1, $2);
if ($cmd eq "cwd" || $cmd eq "cd") {
$curdir = $arg;
} elsif ($cmd eq "unexec" && $arg =~ /^rmdir/) {
- log_warning($line->[0], $line->[1], "use \"\@dirrm\" instead of \"\@unexec rmdir\".");
- } elsif ($cmd eq "exec" || $cmd eq "unexec") {
+ log_warning($line->file, $line->lineno, "use \"\@dirrm\" instead of \"\@unexec rmdir\".");
+ } elsif (($cmd eq "exec" || $cmd eq "unexec")) {
if ($arg =~ /(?:install-info|\$\{INSTALL_INFO\})/) {
- log_warning($line->[0], $line->[1], "\@exec/unexec install-info is deprecated.");
+ log_warning($line->file, $line->lineno, "\@exec/unexec install-info is deprecated.");
} elsif ($arg =~ /ldconfig/ && $arg !~ qr"/usr/bin/true") {
- log_error($line->[0], $line->[1], "ldconfig must be used with \"||/usr/bin/true\".");
+ log_error($line->file, $line->lineno, "ldconfig must be used with \"||/usr/bin/true\".");
}
} elsif ($cmd eq "comment") {
if ($arg =~ /^$regex_rcsidstr$/) {
@@ -621,60 +671,60 @@ sub checkfile_PLIST($) {
} elsif ($cmd eq "dirrm" || $cmd eq "option") {
# no check made
} elsif ($cmd eq "mode" || $cmd eq "owner" || $cmd eq "group") {
- log_warning($line->[0], $line->[1], "\"\@mode/owner/group\" are deprecated, please use chmod/".
+ log_warning($line->file, $line->lineno, "\"\@mode/owner/group\" are deprecated, please use chmod/".
"chown/chgrp in the pkg Makefile and let tar do the rest.");
} else {
- log_warning($line->[0], $line->[1], "unknown PLIST directive \"\@$cmd\"");
+ log_warning($line->file, $line->lineno, "unknown PLIST directive \"\@$cmd\"");
}
next line;
}
- if ($line->[2] =~ /^\//) {
- log_error($line->[0], $line->[1], "use of full pathname disallowed.");
+ if ($line->text =~ /^\//) {
+ log_error($line->file, $line->lineno, "use of full pathname disallowed.");
}
- if ($line->[2] =~ /^doc/) {
- log_error($line->[0], $line->[1], "documentation must be installed under share/doc, not doc.");
+ if ($line->text =~ /^doc/) {
+ log_error($line->file, $line->lineno, "documentation must be installed under share/doc, not doc.");
}
- if ($line->[2] =~ /^etc/ && $line->[2] !~ /^etc\/rc.d/) {
- log_error($line->[0], $line->[1], "configuration files must not be ".
+ if ($line->text =~ /^etc/ && $line->text !~ /^etc\/rc.d/) {
+ log_error($line->file, $line->lineno, "configuration files must not be ".
"registered in the PLIST (don't you use the ".
"PKG_SYSCONFDIR framework?)");
}
- if ($line->[2] =~ /^etc\/rc\.d/) {
- log_error($line->[0], $line->[1], "RCD_SCRIPTS must not be ".
+ if ($line->text =~ /^etc\/rc\.d/) {
+ log_error($line->file, $line->lineno, "RCD_SCRIPTS must not be ".
"registered in the PLIST (don't you use the ".
"RCD_SCRIPTS framework?)");
}
- if ($line->[2] =~ /^info\/dir$/) {
- log_error($line->[0], $line->[1], "\"info/dir\" should not be listed in ".
+ if ($line->text =~ /^info\/dir$/) {
+ log_error($line->file, $line->lineno, "\"info/dir\" should not be listed in ".
"$file. use install-info to add/remove an entry.");
}
- if ($line->[2] =~ /^lib\/locale/) {
- log_error($line->[0], $line->[1], "\"lib/locale\" should not be listed ".
+ if ($line->text =~ /^lib\/locale/) {
+ log_error($line->file, $line->lineno, "\"lib/locale\" should not be listed ".
"in $file. Use \${PKGLOCALEDIR}/locale and set USE_PKGLOCALEDIR instead.");
}
- if ($line->[2] =~ /^share\/locale/) {
- log_warning($line->[0], $line->[1], "use of \"share/locale\" in $file is ".
+ if ($line->text =~ /^share\/locale/) {
+ log_warning($line->file, $line->lineno, "use of \"share/locale\" in $file is ".
"deprecated. Use \${PKGLOCALEDIR}/locale and set USE_PKGLOCALEDIR instead.");
}
- if ($line->[2] =~ /\${PKGLOCALEDIR}/ && $seen_USE_BUILDLINK3 && !$seen_USE_PKGLOCALEDIR) {
- log_warning($line->[0], $line->[1], "PLIST contains \${PKGLOCALEDIR}, ".
+ if ($line->text =~ /\${PKGLOCALEDIR}/ && $seen_USE_BUILDLINK3 && !$seen_USE_PKGLOCALEDIR) {
+ log_warning($line->file, $line->lineno, "PLIST contains \${PKGLOCALEDIR}, ".
"but USE_PKGLOCALEDIR was not found.");
}
if ($curdir !~ m:^$conf_localbase: && $curdir !~ m:^/usr/X11R6:) {
- log_warning($line->[0], $line->[1], "installing to directory $curdir discouraged. could you please avoid it?");
+ log_warning($line->file, $line->lineno, "installing to directory $curdir discouraged. could you please avoid it?");
}
- if ("$curdir/$line->[2]" =~ m:^$conf_localbase/share/doc:) {
- log_info($line->[0], $line->[1], "seen installation to share/doc ($curdir/$line).");
+ if ("$curdir/$line->text" =~ m:^$conf_localbase/share/doc:) {
+ log_info($line->file, $line->lineno, "seen installation to share/doc ($curdir/$line).");
}
}
@@ -746,11 +796,11 @@ sub check_for_multiple_patches($) {
$files_in_patch = 0;
$patch_state = "";
foreach my $line (@$lines) {
- if (index($line->[2], "--- ") == 0 && $line->[2] !~ qr"^--- \d+(?:,\d+|) ----$") {
+ if (index($line->text, "--- ") == 0 && $line->text !~ qr"^--- \d+(?:,\d+|) ----$") {
$line_type = "-";
- } elsif (index($line->[2], "*** ") == 0 && $line->[2] !~ qr"^\*\*\* \d+(?:,\d+|) \*\*\*\*$") {
+ } elsif (index($line->text, "*** ") == 0 && $line->text !~ qr"^\*\*\* \d+(?:,\d+|) \*\*\*\*$") {
$line_type = "*";
- } elsif (index($line->[2], "+++ ") == 0) {
+ } elsif (index($line->text, "+++ ") == 0) {
$line_type = "+";
} else {
$line_type = "";
@@ -761,14 +811,14 @@ sub check_for_multiple_patches($) {
$files_in_patch++;
$patch_state = "";
} else {
- log_warning($line->[0], $line->[1], "unknown patch format (might be an internal error)");
+ log_warning($line->file, $line->lineno, "unknown patch format (might be an internal error)");
}
} elsif ($patch_state eq "-") {
if ($line_type eq "+") {
$files_in_patch++;
$patch_state = "";
} else {
- log_warning($line->[0], $line->[1], "unknown patch format (might be an internal error)");
+ log_warning($line->file, $line->lineno, "unknown patch format (might be an internal error)");
}
} elsif ($patch_state eq "") {
$patch_state = $line_type;
@@ -776,9 +826,9 @@ sub check_for_multiple_patches($) {
}
if ($files_in_patch > 1) {
- log_warning($lines->[0]->[0], NO_LINE_NUMBER, "contains patches for $files_in_patch files, should be only one");
+ log_warning($lines->[0]->file, NO_LINE_NUMBER, "contains patches for $files_in_patch files, should be only one");
} elsif ($files_in_patch == 0) {
- log_warning($lines->[0]->[0], NO_LINE_NUMBER, "contains no patch");
+ log_warning($lines->[0]->file, NO_LINE_NUMBER, "contains no patch");
}
return true;
}
@@ -802,13 +852,13 @@ sub checkfile_patches_patch($) {
if (scalar(@$lines) == 0) {
log_error($fname, NO_LINE_NUMBER, "Empty patch file.");
return false;
- } elsif ($lines->[0]->[2] !~ /^$regex_rcsidstr$/) {
- log_error($lines->[0]->[0], $lines->[0]->[1], "Expected RCS tag \"\$$conf_rcsidstr\$\" (and nothing more) here.");
+ } elsif ($lines->[0]->text !~ /^$regex_rcsidstr$/) {
+ log_error($lines->[0]->file, $lines->[0]->lineno, "Expected RCS tag \"\$$conf_rcsidstr\$\" (and nothing more) here.");
}
foreach my $line (@$lines[1..scalar(@$lines)-1]) {
- if ($opt_committer && $line->[2] =~ /$regex_known_rcs_tag/) {
- log_warning($line->[0], $line->[1], "Possible RCS tag \"\$$1\$\".");
+ if ($opt_committer && $line->text =~ /$regex_known_rcs_tag/) {
+ log_warning($line->file, $line->lineno, "Possible RCS tag \"\$$1\$\". Use binary mode (-ko) on commit/import.");
}
}
@@ -1835,10 +1885,10 @@ sub category_check() {
log_error($file, NO_LINE_NUMBER, "may not be empty.");
return false;
}
- if ($lines->[0]->[2] =~ qr"^# $regex_rcsidstr$") {
- log_info($lines->[0]->[0], $lines->[0]->[1], "RCS Id tag found.");
- } elsif (scalar(@$lines) > 1 && $lines->[1]->[2] =~ qr"^# $regex_rcsidstr$") {
- log_info($lines->[1]->[0], $lines->[1]->[1], "RCS Id tag found.");
+ if ($lines->[0]->text =~ qr"^# $regex_rcsidstr$") {
+ log_info($lines->[0]->file, $lines->[0]->lineno, "RCS Id tag found.");
+ } elsif (scalar(@$lines) > 1 && $lines->[1]->text =~ qr"^# $regex_rcsidstr$") {
+ log_info($lines->[1]->file, $lines->[1]->lineno, "RCS Id tag found.");
} else {
log_error($file, NO_LINE_NUMBER, "No RCS Id tag found.");
}
@@ -1847,31 +1897,31 @@ sub category_check() {
my ($first, $last_subdir, $comment_seen) = (true, undef, false);
foreach my $line (@$lines) {
- if ($line->[2] =~ qr"^(#?)SUBDIR(.*?)=\s*(\S+)\s*(?:#\s*(.*?)\s*|)$") {
+ if ($line->text =~ qr"^(#?)SUBDIR(.*?)=\s*(\S+)\s*(?:#\s*(.*?)\s*|)$") {
my ($comment_flag, $operator, $subdir, $comment) = ($1, $2, $3, $4);
if ($comment_flag eq "#") {
if (defined($comment) && $comment eq "") {
- log_warning($line->[0], $line->[1], "$subdir commented out without giving a reason.");
+ log_warning($line->file, $line->lineno, "$subdir commented out without giving a reason.");
}
push(@makefile_subdirs, $subdir);
} elsif ($first) {
$first = false;
if ($operator ne "" && $operator ne "+") {
- log_error($line->[0], $line->[1], "SUBDIR= or SUBDIR+= expected.");
+ log_error($line->file, $line->lineno, "SUBDIR= or SUBDIR+= expected.");
}
push(@makefile_subdirs, $subdir);
$last_subdir = $subdir;
} else {
if ($operator ne "+") {
- log_error($line->[0], $line->[1], "SUBDIR+= expected.");
+ log_error($line->file, $line->lineno, "SUBDIR+= expected.");
}
push(@makefile_subdirs, $subdir);
if ($last_subdir ge $subdir) {
- log_error($line->[0], $line->[1], "$subdir should come before $last_subdir.");
+ log_error($line->file, $line->lineno, "$subdir should come before $last_subdir.");
}
$last_subdir = $subdir;
}
- } elsif ($line->[2] =~ qr"^COMMENT\s*=\s*([^#]*?)") {
+ } elsif ($line->text =~ qr"^COMMENT\s*=\s*([^#]*?)") {
my ($comment) = ($1);
$comment_seen = true;
}