Skip to main content

Measure Halstead code complexity

Project description

halstead-complexity

A command-line tool for measuring Halstead complexity metrics in source code. Supports Python and JavaScript out of the box, with the ability to add support for any language via tree-sitter grammars.

Quick Start

# Install the tool
pip install halstead-complexity[python]

# Initialize a config file
hc config init

# Analyze a file or directory
hc analyze <path>

# Show each token counted
hc analyze <path> --tokens

# Other options
hc analyze <path> --hal # Only Halstead metrics
hc analyze <path> --raw # Only raw metrics
hc analyze <path> --silence # Only output success message
hc analyze <path> -o report.csv # Write report to file (txt or csv)
hc analyze --help # Show all options

Installation

pip install halstead-complexity

or

uv add halstead-complexity

Language Support

The tool does not come with any language dependencies by default. To add support for other languages, you'll need to install the corresponding tree-sitter grammar packages (see Adding Support for More Languages below).

However, the default configuration includes support for Python and JavaScript. If that is all you need, you can simply install the package with the optional dependencies (python, javascript, all):

pip install halstead-complexity[python]

Metrics

Raw Metrics

  • LOC: Total number of lines of code
  • LLOC: Number of logical lines of code (each contains exactly one statement)
  • SLOC: Number of source lines of code
  • Comments: Number of comment lines
  • Multi-lines: Number of lines representing multi-line delimiters
  • Blank lines: Number of blank or whitespace-only lines

The equation SLOC + Multi-lines + Comments + Blank lines = LOC should always hold.

Halstead Metrics

  • η₁: Total number of distinct operators
  • η₂: Total number of distinct operands
  • N₁: Total number of operators
  • N₂: Total number of operands
  • Vocabulary: η = η₁ + η₂
  • Length: N = N₁ + N₂
  • Volume: V = N × log₂(η)
  • Difficulty: D = (η₁ / 2) × (N₂ / η₂)
  • Effort: E = D × V
  • Time: T = E / 18 seconds
  • Delivered Bugs: B = V / 3000

Configuration

The tool uses a hierarchical configuration system with three levels of precedence:

  1. Default config - Built-in configuration (lowest precedence)
  2. Global config - User-wide configuration at ~/.config/halstead-complexity/config.json
  3. Local config - Project-specific configuration at ./hc_config.json (highest precedence)

Creating a Configuration File

Initialize a new configuration file:

# Create a local config in the current directory
hc config init --local

# Create a global config for all projects
hc config init --global

Configuration Structure

The configuration file is a JSON file with the following structure (see default config):

{
  "default_language": "python",
  "braces_single_operator": false,
  "template_literal_single_operand": false,
  "languages": {
    "python": {
      "comment": ["#"],
      "extensions": [".py"],
      "excluded": ["__pycache__", ".pytest_cache", ".venv"],
      "statement_types": [...],
      "operand_types": [...],
      "keywords": [...],
      "symbols": [...],
      "multi_word_operators": ["is not", "not in"],
      "multi_line_delimiters": [
        {"start": "\"\"\"", "end": "\"\"\""},
        {"start": "'''", "end": "'''"}
      ]
    }
  }
}

Configuration Fields

  • default_language: The default language to use when analyzing files. Used for directory analysis. Single file analysis will use the file extension to determine the language. Must be one of the languages defined in languages.

  • braces_single_operator: Whether to treat opening/closing braces as single operators (default: false)

    true: {} is a single operator

    false: { and } are separate operators

  • template_literal_single_operand: Whether to treat template literals as single operands (default: false)

    true: f"{n} is odd." is a single operand

    false: " is odd." is an operand and { } are operators

  • languages: Language-specific configurations

Each language configuration includes:

  • comment: Single-line comment syntax (e.g., ["#"] for Python, ["//"] for JavaScript)
  • extensions: File extensions for this language (e.g., [".py"], [".js", ".mjs"])
  • excluded: Directories to exclude when analyzing (e.g., ["__pycache__", "node_modules"])
  • statement_types: Tree-sitter node types that represent statements
  • operand_types: Tree-sitter node types that represent operands (identifiers, literals, etc.)
  • keywords: Language keywords that are considered operators
  • symbols: Operator symbols (e.g., +, -, ==, !=)
  • multi_word_operators: Multi-word operators (e.g., "is not", "not in")
  • multi_line_delimiters: Multi-line comment/string delimiters

Adding Support for More Languages

The tool uses tree-sitter for parsing source code, which means you can add support for any language that has a tree-sitter grammar.

Step 1: Install the Tree-Sitter Grammar

First, install the Python bindings for the tree-sitter grammar. Most languages have packages available on PyPI with the naming convention tree-sitter-{language}.

# Example: Adding Rust support
pip install tree-sitter-rust

Find tree-sitter grammars by:

  • Searching for tree-sitter-{language} on PyPI
  • Checking the list of parsers (all these may not exists on PyPI)

Step 2: Configure the Language

Add a configuration for the new language to your config file:

# Initialize a config file if you haven't already
hc config init

Edit the config file (e.g., hc_config.json) and add your language configuration:

  • See the default config for an example.
  • Check the grammar definition in the tree-sitter repository (e.g., tree-sitter-rust/grammar.js)

CLI Reference

Usage:

$ hc [OPTIONS] COMMAND [ARGS]...

Options:

  • -v, --version: Show the application's version and exit.
  • --install-completion: Install completion for the current shell.
  • --show-completion: Show completion for the current shell, to copy it or customize the installation.
  • --help: Show this message and exit.

Commands:

  • analyze: Analyze source code file or directory for complexity metrics.
  • config: Manage Halstead Complexity config files.

hc analyze

Analyze source code file or directory for complexity metrics.

Usage:

$ hc analyze [OPTIONS] PATH

Arguments:

  • PATH: Path to a file or directory to analyze [required]

Options:

  • --hal: Only show Halstead metrics
  • --raw: Only show raw metrics
  • --tokens: Show operators and operands
  • --silence: Only output success message
  • -o, --output TEXT: Write report to file
  • -c, --config TEXT: Path to config file
  • --help: Show this message and exit.

hc config

Manage Halstead Complexity config files.

Usage:

$ hc config [OPTIONS] COMMAND [ARGS]...

Options:

  • --help: Show this message and exit.

Commands:

  • init: Initialize a new config file.
  • get: Get a config value.
  • set: Set a config value.
  • list: List all config values.
  • path: Show the path to the config file.

hc config init

Initialize a new config file.

Usage:

$ hc config init [OPTIONS]

Options:

  • --local: Use the config file in the current working directory.
  • --global: Use the global config file.
  • --help: Show this message and exit.

hc config get

Get a config value.

Usage:

$ hc config get [OPTIONS] KEY

Arguments:

  • KEY: [required]

Options:

  • --local: Use the config file in the current working directory.
  • --global: Use the global config file.
  • --help: Show this message and exit.

hc config set

Set a config value.

Usage:

$ hc config set [OPTIONS] KEY VALUE

Arguments:

  • KEY: [required]
  • VALUE: [required]

Options:

  • --local: Use the config file in the current working directory.
  • --global: Use the global config file.
  • --help: Show this message and exit.

hc config list

List all config values.

Usage:

$ hc config list [OPTIONS]

Options:

  • --local: Use the config file in the current working directory.
  • --global: Use the global config file.
  • --help: Show this message and exit.

hc config path

Show the path to the config file.

Usage:

$ hc config path [OPTIONS]

Options:

  • --local: Use the config file in the current working directory.
  • --global: Use the global config file.
  • --help: Show this message and exit.

Contributing

  • More default language support is welcome. Please add the configuration to the default config and include the optional dependency in the pyproject.toml.
  • For now the project is focused on Halstead complexity. But it could potentially be expanded to include other metrics such as cyclomatic complexity and maintainability index.

Credits

This project was inspired by:

License

Licensed under the MIT license

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

halstead_complexity-0.1.0.tar.gz (50.2 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

halstead_complexity-0.1.0-py3-none-any.whl (24.2 kB view details)

Uploaded Python 3

File details

Details for the file halstead_complexity-0.1.0.tar.gz.

File metadata

  • Download URL: halstead_complexity-0.1.0.tar.gz
  • Upload date:
  • Size: 50.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: uv/0.9.2

File hashes

Hashes for halstead_complexity-0.1.0.tar.gz
Algorithm Hash digest
SHA256 48d3c29fe82427859a97715ee0eceed27cf46dbf21e25ecee8f7da04a6fc6b4c
MD5 2e252a67171067f58acbfb8d906c4e9f
BLAKE2b-256 7af5ec87639cab796b8ec8b66e88a6fb6e13cfe0df36288ffecaa6796999cacf

See more details on using hashes here.

File details

Details for the file halstead_complexity-0.1.0-py3-none-any.whl.

File metadata

File hashes

Hashes for halstead_complexity-0.1.0-py3-none-any.whl
Algorithm Hash digest
SHA256 ac324bee8237847e51a3b02bde6a0222e655eba4ea5e7fbed8ce761dd0d8a778
MD5 ac72683842567852dcea1227765d4910
BLAKE2b-256 c65dbc1d7e4b218038c5e8a99d82e57476d4e28053b22b7c8dd56d4074387017

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page