Warning
This is a University Research Project and SHOULD NOT BE USED ON PRODUCTION ENVIRONMENTS. The goal is to determine the feasibility, performance and developer experience of Rust Programming Language for implementing static code analyzers for Ballerina Language.
- Benchmarks
- Documentation
- Installation
- Usage
- Configuration
- Development environment
- Building
- Debugging
- Contributing
- TODO
- License
Followings are the benchmarks we achieved using the release version 0.3.0:
| Tool | Total Execution Time | Tokens/Second | Performance vs Blazelint |
|---|---|---|---|
| Blazelint | 284ms | 2,290,000 | Baseline (1×) |
| Ballerina Scan | 8m 04.41s (484.41s) | 1,341 | 1,705× slower |
| ESLint | 3.6s | 188,900 | 12.7× slower |
| Stage | Execution Time | Percentage of Total | Tokens Processed |
|---|---|---|---|
| Lexical Analysis | 80.75ms | 26.9% | 650,306 |
| Parsing | 187.90ms | 62.6% | 650,306 |
| Semantic Analysis | 22.87ms | 7.6% | 650,306 |
| Linting Rules | 8.87ms | 3.0% | 650,306 |
| Total | 300.39ms | 100.0% | 650,306 |
- Grammar Coverage: Blazelint currently implements approximately 23% of the full Ballerina grammar. The benchmarks reflect this partial implementation, and performance numbers should be considered with this limitation in mind.
- Lexer Scalability: The lexer uses a switch-case dispatch mechanism, ensuring constant time complexity per character. Adding new lexemes will not significantly impact performance.
- Parser Scalability: Uses a recursive descent parser is designed for modular expansion. While adding new grammar rules increases the depth of recursive calls, the architecture supports efficient scaling with minimal overhead for additional rules.
- BNF Grammar for Ballerina Subset
- Software Requirement Specification (SRS)
- Pipeline overview
- Quick Reference
- Implementation Notes
Install the latest published version from crates.io:
cargo install blazelintPre-build binaries are available for Linux from the latest GitHub release.
Windows and MacOS binaries will be added in a later release.
Analyze a Ballerina source file by passing its path to blazelint:
blazelint path/to/file.balNote
Use the limited subset documented in the BNF when defining Ballerina syntax to be linted.
The tool prints the detected diagnostics if there is any and exits with a non-zero status or exits with a zero status with no prints to stdout if the passed file is clean.
Running from a checked-out repository is also supported:
cargo run -- path/to/file.balNote
cargo run builds and executes an unoptimized build (for debug requirements). Always use cargo build --release for any benchmark or observations on performance.
For a quick smoke test, you can reuse the sample program in tests/test-bal-files/:
blazelint tests/test-bal-files/simple_errors.balBlazelint supports timing analysis for each pipeline stage. Use the following flags:
--timing: Displays the total time taken by each pipeline stage (lexing, parsing, semantic analysis, linting).--detailed-timing: Provides a detailed breakdown, including per-rule linting durations.
Example:
blazelint --timing path/to/file.balBlazelint looks for a .blazerc configuration file in the current directory or any parent directory. The configuration uses TOML format:
# .blazerc - Blazelint Configuration File
[rules]
# Naming convention rules
camel-case = "error" # Enforces camelCase for variables/functions
constant-case = "warn" # Enforces SCREAMING_SNAKE_CASE for constants
# Code style rules
line-length = "warn" # Limits line length
max-function-length = "error" # Limits function body length
missing-return = "error" # Ensures functions have return statements
unused-variables = "warn" # Detects unused variable declarations
# Disable specific rules
some-rule = "off"
[settings]
max-line-length = 120 # Maximum characters per line
max-function-length = 50 # Maximum lines in function bodyEach rule can be configured with one of these severity levels:
"error"- Causes build failure (non-zero exit code)"warn"- Shows warnings but allows build to succeed"info"- Shows informational messages"off"- Disables the rule completely
| Rule | Description | Default Severity | Settings |
|---|---|---|---|
camel-case |
Enforces camelCase naming for variables and functions | error |
None |
constant-case |
Enforces SCREAMING_SNAKE_CASE for constants | warn |
None |
line-length |
Limits line length | warn |
max-line-length |
max-function-length |
Limits function body length | warn |
max-function-length |
missing-return |
Ensures functions have return statements | error |
None |
unused-variables |
Detects unused variable declarations | warn |
None |
Blazelint searches for .blazerc files in this order:
- Current directory:
./.blazerc - Parent directories: Walks up the directory tree looking for
.blazerc - Default configuration: Uses built-in defaults if no file found
The rule engine features:
- Dynamic Rule Loading: Only enabled rules are executed
- Configurable Severity: Each rule respects configured severity levels
- Caching: Configuration is cached for performance
- Extensible Design: New rules can be added easily
A pre-configured Dev Container is available that can be used to investigate, develop or debug the program without installing anything on the host machine.
It can be launched and used fully remotely inside a browser using GitHub codespaces, or locally using Visual Studio Code.
- Click Code → Create codespace from the GitHub UI.
- Wait for the Codespace to provision (first run will take some significant time).
- Start Developing!
- Install the Dev Containers extension.
- Clone this repository and open it in VS Code.
- Run the Dev Containers: Reopen in Container command.
- Wait till the container spins up.
- Start Developing!
The container comes with:
- Rust toolchain
- Typst CLI for building the SRS
- Ballerina runtime
- Extensions for Language Servers, syntax highlighting and debugging support
- Common utilities (zsh, GitHub CLI, git, etc.)
The project uses the following key dependencies:
- Core: Standard library only for main linting logic
- Configuration:
serde,tomlfor config file parsing - Utilities:
once_cell,thiserrorfor error handling and caching - Testing:
assert_cmd,tempfilefor integration tests
- Git 2.51.0 or newer
- Rust Toolchain 1.86.0 or newer (Get it here)
-
Create a fork and clone to local:
git clone https://github.com/<your-profile-name>/Blazelint.git
-
cdinto the directory:cd Blazelint -
Build with cargo:
cargo build --release
- Build requirements stated here.
- Visual Studio Code IDE by Microsoft
- Rust Analyzer extension by rust-lang.org
- CodeLLDB extension by Vadim Chugunov
- Ballerina toolchain and IDE extension (optional - for testing or writing ballerina codes)
- You can adjust the
tests/test-bal-files/files if you need to debug a specific diagnostic. - Create a
.blazercconfig file to test configuration changes. - Set breakpoints as needed.
- Click on Run and Debug from the main method or use
ctrl+shift+Dto jump to debug menu.
Note
It is possible to debug with any IDE including Neovim, Emacs and etc but we recommend Visual Studio Code for easier setup.
-
Changes should be developed and push to following branches based on the area of the feature.
- feature/linter-core: Changes to the linter engine (lexer, parser, semantic analyzer and BNF document).
- feature/rule-engine: Changes to rule engine, configuration system, and linter rules.
- ci/cd: Changes related to continous integration and deployments.
- docs: Changes related to documentation.
-
Run all formatter, lint, and test checks locally before opening a pull request:
bash scripts/check.sh
This project is licensed under the MIT License.
