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:
- Default config - Built-in configuration (lowest precedence)
- Global config - User-wide configuration at
~/.config/halstead-complexity/config.json - 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 inlanguages. -
braces_single_operator: Whether to treat opening/closing braces as single operators (default:false)true:
{}is a single operatorfalse:
{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 operandfalse:
" 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 statementsoperand_types: Tree-sitter node types that represent operands (identifiers, literals, etc.)keywords: Language keywords that are considered operatorssymbols: 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
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
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
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
48d3c29fe82427859a97715ee0eceed27cf46dbf21e25ecee8f7da04a6fc6b4c
|
|
| MD5 |
2e252a67171067f58acbfb8d906c4e9f
|
|
| BLAKE2b-256 |
7af5ec87639cab796b8ec8b66e88a6fb6e13cfe0df36288ffecaa6796999cacf
|
File details
Details for the file halstead_complexity-0.1.0-py3-none-any.whl.
File metadata
- Download URL: halstead_complexity-0.1.0-py3-none-any.whl
- Upload date:
- Size: 24.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: uv/0.9.2
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
ac324bee8237847e51a3b02bde6a0222e655eba4ea5e7fbed8ce761dd0d8a778
|
|
| MD5 |
ac72683842567852dcea1227765d4910
|
|
| BLAKE2b-256 |
c65dbc1d7e4b218038c5e8a99d82e57476d4e28053b22b7c8dd56d4074387017
|