NAME
Perl::Critic::Policy::CodeLayout::RequireFinalSemicolon - require a semicolon at the end of code blocks
DESCRIPTION
This policy is part of the Perl::Critic::Pulp
add-on. It asks you to put a semicolon ;
on the final statement of a subroutine or block.
sub foo {
do_something(); # ok
}
sub bar {
do_something() # bad
}
The idea is that if you add more code you don't have to notice the previous line needs a terminator. It's also more like the C language, if you consider that a virtue.
This is only a matter of style since the code runs the same either way, and on that basis this policy is low severity and under the "cosmetic" theme (see "POLICY THEMES" in Perl::Critic).
Same Line Closing Brace
By default (see "CONFIGURATION" below) a semicolon is not required when the closing brace is on the same line as the last statement. This is good for constants and one-liners.
sub foo { 'my-constant-value' } # ok
sub square { return $_[0] ** 2 } # ok
Final Value Expression
A semicolon is not required in places where the last statement is an expression giving a value.
map { some_thing();
$_+123 # ok
} @values;
do {
foo();
1+2+3 # ok
}
This currently means
do grep map sort # builtins
reduce any all none notall first # List::Util
pairfirst pairgrep pairmap
mapp map_pairwise grepp grep_pairwise # List::Pairwise
firstp first_pairwise lastp last_pairwise
The module functions are always treated as expressions. There's no check for whether the respective module is actually in use. Fully qualified names like List::Util::first
are recognised too.
do {} while
or do {} until
loops are ordinary blocks, not expression blocks, so still require a semicolon on the last statement inside.
do {
foo() # bad
} until ($condition);
The last statement of a sub{}
is not considered an expression. Perhaps there could be an option to excuse all one-statement subs or even all subs and have the policy just for nested code and control blocks. For now the suggestion is that if a sub is big enough to need a separate line for its result expression then write an actual return
statement for maximum clarity.
Try/Catch Blocks
The Try
, TryCatch
and Syntax::Feature::Try
modules all add try
block forms. These statements don't require a terminating semicolon (the same as an if
doesn't).
use TryCatch;
sub foo {
try {
attempt_something();
} catch {
error_recovery();
} # ok, no semi required here for TryCatch
}
The insides of the try
and catch
are treated the same as other blocks. But the try
statement itself doesn't require a semicolon. (See policy ValuesAndExpressions::ProhibitNullStatements
to notice one added unnecessarily.)
For reference, PPI
doesn't know try
/catch
specifically, so when they don't have a final semicolon the next statement runs together and the nature of those parts might be lost. This could upset things like recognition of for
loops and could potentially make some perlcritic reports go wrong.
The try
/catch
block exemption here is only for the modules with this block syntax. There are other try modules such as Try::Tiny
and friends where a final semicolon is normal and necessary if more code follows (because their try
and catch
are ordinary function calls prototyped to take code blocks).
use Try::Tiny;
sub foo {
try {
attempt_something();
} catch {
error_recovery();
} # bad, semi required here for Try::Tiny
}
Disabling
If you don't care about this you can always disable from your .perlcriticrc file in the usual way (see "CONFIGURATION" in Perl::Critic),
[-CodeLayout::RequireFinalSemicolon]
CONFIGURATION
except_same_line
(boolean, default true)-
If true (the default) then don't demand a semicolon if the closing brace is on the same line as the final statement.
sub foo { return 123 } # ok if "except_same_line=yes" # bad if "except_same_line=no"
except_expression_blocks
(boolean, default true)-
If true (the default) then don't demand a semicolon at the end of an expression block, as described under "Final Value Expression" above.
# ok under "except_expression_blocks=yes" # bad under "except_expression_blocks=no" do { 1+2+3 } map { $_+1 } @array grep {defined} @x
The statements and functions for this exception are currently hard coded. Maybe in the future they could be configurable, though multi-line expressions in this sort of thing tends to be unusual anyway. (See policy
BuiltinFunctions::RequireSimpleSortBlock
to demandsort
is only one line.)
BUGS
It's very difficult to distinguish a code block from an anonymous hashref constructor if there might be a function prototype in force, eg.
foo { abc => 123 }; # hash ref normally
# code block if foo() has prototype
PPI
tends to assume code. RequireFinalSemicolon
currently assumes hashref so as to avoid false violations. Any try
, catch
or finally
are presumed to be code blocks (the various Try modules). Perhaps other common or particular functions or syntax with code blocks could be recognised. In general this sort of ambiguity is another good reason to avoid function prototypes.
SEE ALSO
Perl::Critic::Pulp, Perl::Critic, Perl::Critic::Policy::CodeLayout::RequireTrailingCommas, Perl::Critic::Policy::CodeLayout::RequireTrailingCommaAtNewline, Perl::Critic::Policy::Subroutines::RequireFinalReturn, Perl::Critic::Policy::ValuesAndExpressions::ProhibitNullStatements, Perl::Critic::Policy::BuiltinFunctions::RequireSimpleSortBlock
List::Util, List::Pairwise, Try, TryCatch, Syntax::Feature::Try
HOME PAGE
http://user42.tuxfamily.org/perl-critic-pulp/index.html
COPYRIGHT
Copyright 2010, 2011, 2012, 2013, 2014, 2015, 2016, 2017 Kevin Ryde
Perl-Critic-Pulp is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3, or (at your option) any later version.
Perl-Critic-Pulp is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License along with Perl-Critic-Pulp. If not, see <http://www.gnu.org/licenses>.