You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
108 lines
2.1 KiB
108 lines
2.1 KiB
#!/usr/bin/perl |
|
|
|
use strict; |
|
use warnings; |
|
|
|
my %SECTIONS; |
|
{ |
|
my $order = 0; |
|
%SECTIONS = ( |
|
'NAME' => { |
|
required => 1, |
|
order => $order++, |
|
}, |
|
'SYNOPSIS' => { |
|
required => 1, |
|
order => $order++, |
|
}, |
|
'DESCRIPTION' => { |
|
required => 1, |
|
order => $order++, |
|
}, |
|
'OPTIONS' => { |
|
order => $order++, |
|
required => 0, |
|
}, |
|
'CONFIGURATION' => { |
|
order => $order++, |
|
}, |
|
'BUGS' => { |
|
order => $order++, |
|
}, |
|
'SEE ALSO' => { |
|
order => $order++, |
|
}, |
|
'FILE FORMAT' => { |
|
order => $order++, |
|
}, |
|
'GIT' => { |
|
required => 1, |
|
order => $order++, |
|
}, |
|
); |
|
} |
|
my $SECTION_RX = do { |
|
my ($names) = join "|", keys %SECTIONS; |
|
qr/^($names)$/s; |
|
}; |
|
|
|
my $exit_code = 0; |
|
sub report { |
|
my ($msg) = @_; |
|
print STDERR "$ARGV:$.: $msg\n"; |
|
$exit_code = 1; |
|
} |
|
|
|
my $last_was_section; |
|
my @actual_order; |
|
while (my $line = <>) { |
|
chomp $line; |
|
if ($line =~ $SECTION_RX) { |
|
push @actual_order => $line; |
|
$last_was_section = 1; |
|
# Have no "last" section yet, processing NAME |
|
next if @actual_order == 1; |
|
|
|
my @expected_order = sort { |
|
$SECTIONS{$a}->{order} <=> $SECTIONS{$b}->{order} |
|
} @actual_order; |
|
|
|
my $expected_last = $expected_order[-2]; |
|
my $actual_last = $actual_order[-2]; |
|
if ($actual_last ne $expected_last) { |
|
report("section '$line' incorrectly ordered, comes after '$actual_last'"); |
|
} |
|
next; |
|
} |
|
if ($last_was_section) { |
|
my $last_section = $actual_order[-1]; |
|
if (length $last_section ne length $line) { |
|
report("dashes under '$last_section' should match its length!"); |
|
} |
|
if ($line !~ /^-+$/) { |
|
report("dashes under '$last_section' should be '-' dashes!"); |
|
} |
|
$last_was_section = 0; |
|
} |
|
|
|
if (eof) { |
|
# We have both a hash and an array to consider, for |
|
# convenience |
|
my %actual_sections; |
|
@actual_sections{@actual_order} = (); |
|
|
|
for my $section (sort keys %SECTIONS) { |
|
next if !$SECTIONS{$section}->{required} or exists $actual_sections{$section}; |
|
report("has no required '$section' section!"); |
|
} |
|
|
|
# Reset per-file state |
|
{ |
|
@actual_order = (); |
|
# this resets our $. for each file |
|
close ARGV; |
|
} |
|
} |
|
} |
|
|
|
exit $exit_code;
|
|
|