/python-abc

A python implementation of the ABC sofware metric.

Primary LanguagePythonThe UnlicenseUnlicense

Python ABC

Python 3.9.5

A python implementation of the ABC Software metric:

The ABC software metric was introduced by Jerry Fitzpatrick in 1997 to overcome the drawbacks of the LOC. The metric defines an ABC score as a triplet of values that represent the size of a set of source code statements. An ABC score is calculated by counting the number of assignments (A), number of branches (B), and number of conditionals (C) in a program. ABC score can be applied to individual methods, functions, classes, modules or files within a program.

Fitzpatrick's original paper is, at the time of writing, only available via the Wayback Machine, so a copy of it is included in this repo as well.

The paper lists the counting rules for C, C++ and Java, so here are the rules this repo uses for Python:

  • Add one to the assignment count when:
    • Occurrence of an assignment operator (excluding default parameter assignments).
  • Add one to the branch count when:
    • Occurrence of a function call/await or a class method call/await.
    • Occurrence of a class instantiation.
  • Add one to the condition count when:
    • Occurrence of a conditional operator.
    • Occurrence of the following keywords: else, elif, except.
    • Occurrence of an assert statement without a conditional operator.

Usage

Install the requirements in your virtual environment of choice, then you can see the command line arguments that are available:

$ python -m python_abc --help
usage: python_abc [-h] [--debug DEBUG] [--sort SORT] [--verbose VERBOSE] path

A python implementation of the ABC Software metric: https://en.wikipedia.org/wiki/ABC_Software_Metric

positional arguments:
  path               path to directory or file

optional arguments:
  -h, --help         show this help message and exit
  --debug DEBUG      display AST output for each element in the parsed tree
  --sort SORT        sort files from highest to lowest magnitude
  --verbose VERBOSE  display marked-up file

Given file.py that contains the following text:

if a and b:
    print(a)
else:
    print(b)

a = sum(i for i in range(1000) if i % 3 == 0 and i % 5 == 0)

def f(n):
    def inner(n):
        return n ** 2
    if n == 0:
        return 1
    elif n == 1:
        return n
    elif n < 5:
        return (n - 1) ** 2
    return n * pow(inner(n), f(n - 1), n - 3)

You can get the barebones output as follows:

$ python -m python_abc /path/to/file.py
/path/to/file.py         <1, 7, 10> (12.2)

Passing the verbose flag will give more detail:

$ python -m python_abc file.py --verbose=true
cc    | if a and b:
b     |     print(a)
c     | else:
b     |     print(b)
      |
abbcc | a = sum(i for i in range(1000) if i % 3 == 0 and i % 5 == 0)
      |
      | def f(n):
      |     def inner(n):
      |         return n ** 2
c     |     if n == 0:
      |         return 1
cc    |     elif n == 1:
      |         return n
cc    |     elif n < 5:
      |         return (n - 1) ** 2
bbb   |     return n * pow(inner(n), f(n - 1), n - 3)
file.py          <1, 7, 10> (12.2)

If you want to inspect the abstract syntax tree for the file you can pass the debug flag, which will print out each node from the tree and the vector that resulted from it.

The path argument can also be a path to a directory, in which case all Python files in that directory (and its sub-directories) will be scanned, at which point it can be useful to pass the sort flag to rank the files by ABC magnitude:

$ python -m python_abc . --sort
./calculate.py                              <18, 56, 23> (63.2)
./vector.py                                 <12, 23, 11> (28.2)
./main.py                                    <10, 23, 8> (26.3)
./tests/test_vector.py                       <4, 19, 10> (21.8)
./tests/__init__.py                           <4, 12, 1> (12.7)
./tests/test_radon_test_cases.py                <1, 2, 1> (2.4)
./tests/test_calculate_condition.py             <1, 2, 1> (2.4)
./tests/test_calculate_empty.py                 <1, 2, 1> (2.4)
./tests/test_calculate_assignment.py            <1, 2, 1> (2.4)
./tests/test_calculate_branch.py                <1, 2, 1> (2.4)

Finally you can pass a cores argument to tell the library how many CPU cores to use. By default the library will try to use all the cores that are available on your machine.