# NAME

Code::Quality - use static analysis to compute a "code quality" metric for a program

# SYNOPSIS

```perl
use v5.20;
use Code::Quality;
# code to test (required)
my $code = ...;
# reference code to compare against (optional)
my $reference = ...;

my $warnings =
  analyse_code
    code => $code,
    reference => $reference,
    language => 'C';
if (defined $warnings) {
  my $stars = star_rating_of_warnings $warnings;
  say "Program is rated $stars stars"; # 3 is best, 1 is worst
  my @errors = grep { $_->[0] eq 'error' } @$warnings;
  if (@errors > 0) {
    say 'Found ', scalar @errors, ' errors';
    say "First error:  $errors[0][1]";
  }
} else {
  say 'Failed to analyse code';
}
```

# DESCRIPTION

Code::Quality runs a series of tests on a piece of source code to
compute a code quality metric. Each test returns a possibly empty list
of warnings, that is potential issues present in the source code. This
list of warnings can then be turned into a star rating: 3 stars for
good code, 2 stars for acceptable code, and 1 stars for dubious code.

## Warnings

A warning is an arrayref `[type, message, row, column]`, where
the first two entries are mandatory and the last two can be either
both present or both absent.
The type is one of `qw/error warning info/`.

Four-element warnings correspond to ACE code editor annotations.
Two-element warnings apply to the entire document, not a specific
place in the code.

## Tests

A test is a function that takes key-value arguments:

**test\_something**(code => _$code_, language => _$language_, \[reference => _$reference_, formatted\_code => _$formatted_\])

Here _$code_ is the code to be tested, _$language_ is the
programming language, _$reference_ is an optional reference source
code to compare _$code_ against, and _$formatted\_code_ is the
optional result of running _$code_ through a source code formatter.

Each test returns undef if the test failed (for example, if the test
cannot be applied to this programming language), and an arrayref of
warnings otherwise.

Most tests have several configurable parameters, which come from
global variables. The documentation of each test mentions the global
variables that affect its operations. `local` can be used to run a
test with special configuration once, without affecting other code:

```perl
{
  local $Code::Quality::bla_threshold = 5;
  test_bla code => $code, language => 'C';
}
```

### test\_lines

This test counts non-empty lines in both the formatted code and the reference.
If no formatted code is available, the original code is used.
If the code is significantly longer than the reference, it returns a warning.
If the code is much longer, it returns an error.
Otherwise it returns an empty arrayref.

The thresholds for raising a warning/error are available in the source
code, see global variables `@short_code_criteria` and
`@long_code_criteria`.

This test fails if no reference is provided, but is language-agnostic

### test\_clang\_tidy

This test runs the
[clang-tidy](https://clang.llvm.org/extra/clang-tidy/) static analyser
on the code and returns all warnings found.

The clang-tidy checks in use are determined by two global variables,
each of which is a list of globs such as `modernize-*`. The checks in
`@clang_tidy_warnings` produce warnings, while the checks in
`@clang_tidy_errors` produce errors. There is also a hash
`%clang_tidy_check_options` which contains configuration for the
checks. Finally, the path to the clang-tidy executable is
`$clang_tidy_path`, which is initialized by looking in the PATH using
[File::Which](https://metacpan.org/pod/File::Which). Set this variable to undef to disable this test.

This test does not require a reference, but is limited to languages
that clang-tidy understands. This is controlled by the global variable
`%clang_tidy_extension_of_language`, which contains file extensions
for the supported languages.

### test\_lizard

This test runs the [lizard.py](https://github.com/terryyin/lizard)
code complexity analyser on the code, and reports a warning for every
function that has high cyclomatic complexity, or that is too long.

The thresholds that determine whether a warning or an error are raised
are determined by four global variables, `$lizard_warning_loc`,
`$lizard_error_loc`, `$lizard_warning_ccn`, `$lizard_error_ccn`.
Finally, the path to the lizard executable is `$lizard_path`, which
is initialized by looking in the PATH using [File::Which](https://metacpan.org/pod/File::Which). Set this
variable to undef to disable this test.

This test does not require a reference, but is limited to languages
that lizard understands. This is controlled by the global variable
`%lizard_extension_of_language`, which contains file extensions for
the supported languages.

### analyse\_code

**analyse\_code** runs every test above on the code, producing a
combined list of warnings. It fails (returns undef) if all tests fail.
The tests run by **analyse\_code** are those in the global variable
`@all_tests`, which is a list of coderefs.

## Star rating

**star\_rating\_of\_warnings**(_$warnings_) is a subroutine that takes
the output of a test and computes the star rating as an integer. The
rating is undef if the test failed, 1 if the test returned at least
one error, 2 if the test returned at least one warning but no errors,
and 3 otherwise. So a program gets 3 stars if it only raises
informational messages, or no messages at all.

# EXPORT

By default only **analyse\_code** and **star\_rating\_of\_warnings** are exported.

The other tests can be exported on request.

# AUTHOR

Marius Gavrilescu, <marius@ieval.ro>

# COPYRIGHT AND LICENSE

Copyright (C) 2019 by Wellcode PB SRL

Code::Quality is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as
published by the Free Software Foundation, either version 3 of the
License, or (at your option) any later version.

Code::Quality 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
Affero General Public License for more details.

You should have received a copy of the GNU Affero General Public
License along with Code::Quality. If not, see
[https://www.gnu.org/licenses/](https://www.gnu.org/licenses/).