A fast, configurable Fortran formatter with Fypp, Doxygen, and OpenACC/OpenMP support
Project description
ffmt
A fast, configurable Fortran formatter with support for Fypp, Doxygen, and OpenACC/OpenMP directives. Written in Rust. Installable via pip.
Installation
pip install ffmt
Or via Cargo:
cargo install ffmt
Usage
ffmt src/ # format in-place
ffmt --check src/ # CI mode (exit 1 if changes needed)
ffmt --diff src/ # show colored diff
ffmt -j 8 src/ # parallel
cat file.fpp | ffmt - # stdin/stdout
ffmt --range 10:50 file.fpp # format only lines 10-50
What it does
- Indentation -- scope-based indentation for all Fortran constructs
- Whitespace -- consistent spacing around operators, commas, colons, parentheses
- Keywords -- case normalization (
IF->if,ENDDO->end do) - Continuation lines -- normalized with proportional re-indentation
- Preprocessor -- Fypp (
#:if,#:for), OpenACC (!$acc), OpenMP (!$omp) handled correctly
Configuration
Create ffmt.toml or add [tool.ffmt] to pyproject.toml. All options have sensible defaults -- most projects need no config file at all.
General
| Option | Default | Description |
|---|---|---|
indent-width |
4 | Number of spaces per indentation level. |
line-length |
132 | Maximum line length. Code and comments are wrapped at this limit. Set to 1000 to disable wrapping. |
keyword-case |
"lower" |
Case for Fortran keywords: "lower", "upper", or "preserve". |
normalize-keywords |
true | Split compound keywords like enddo into end do. |
named-ends |
true | Add procedure/module name to bare end statements (end subroutine -> end subroutine s_foo). |
align-declarations |
true | Vertically align :: in consecutive declaration blocks. |
unicode-to-ascii |
true | Replace Unicode Greek letters and math symbols with LaTeX equivalents in comments (σ -> \sigma). |
rewrap-comments |
true | Re-wrap long comments at line-length, splitting at word boundaries. Doxygen !> / !! blocks are joined and re-wrapped as units. |
rewrap-code |
true | Re-wrap long code lines at line-length using token-aware splitting (never breaks inside numbers or strings). |
space-after-comment |
true | Ensure a space after ! in regular comments (!comment -> ! comment). Does not affect !$acc, !<, !>, !!, or !&. |
collapse-double-spaces |
true | Collapse runs of multiple spaces to a single space in code (not in strings or comments). |
keyword-paren-space |
true | Add a space between control-flow keywords and ( (if( -> if (). Applies to if, call, allocate, select case, where, etc. |
fypp-list-commas |
true | Normalize comma spacing inside Fypp '[...]' list arguments. |
indent-fypp |
true | Indent Fypp preprocessor blocks (#:if, #:for, #:call). |
indent-module |
true | Indent the body of module and program blocks. |
Whitespace
The [whitespace] section controls spacing around specific operators:
| Option | Default | Description |
|---|---|---|
relational |
true | Space around ==, /=, <, <=, >, >=. |
logical |
true | Space around .and., .or., .not., .eqv., .neqv.. |
plusminus |
true | Space around binary + and -. |
multdiv |
false | Space around * and /. |
power |
false | Space around **. |
assignment |
true | Space around = in assignments. |
pointer |
true | Space around =>. |
concatenation |
true | Space around // (string concatenation). |
declaration |
true | Space around :: in declarations. |
comma |
true | Space after ,. |
slice-colon |
false | Space around : in array slices. |
Files
| Option | Default | Description |
|---|---|---|
extensions |
["fpp", "f90", "F90", "f95", "f03", "F", "F95", "F03"] |
File extensions to format. |
exclude |
[] |
Glob patterns to exclude. |
respect-gitignore |
true | Skip files listed in .gitignore. |
Example
indent-width = 4
line-length = 132
keyword-case = "lower"
normalize-keywords = true
[whitespace]
relational = true
logical = true
plusminus = true
multdiv = false
power = false
assignment = true
declaration = true
comma = true
Preserved as-is
- String literals and inline expressions (
${...}$,@{...}@) - Comment contents and Doxygen alignment (
!<,!>,!!) - Continuation line structure across
#ifdef/#:ifpreprocessor blocks #ifdef/#endifblocks (no indentation change)- Fypp/Python code inside
$:and@:macro invocations
Editor integration
ffmt includes a built-in LSP server for real-time format-on-save:
ffmt --lsp
VS Code
{
"fortran.formatting.formatter": "ffmt",
"fortran.formatting.args": ["--stdin-filepath", "${file}", "-"]
}
Vim/Neovim
autocmd BufWritePost *.fpp,*.f90 silent !ffmt %
" Or: set formatprg=ffmt\ -
Neovim (LSP)
vim.lsp.start({
name = "ffmt",
cmd = { "ffmt", "--lsp" },
root_dir = vim.fs.dirname(vim.fs.find({ "ffmt.toml", "pyproject.toml" }, { upward = true })[1]),
})
CI integration
GitHub Actions
- uses: sbryngelson/ffmt@latest
with:
args: "--check src/"
pre-commit
repos:
- repo: https://github.com/sbryngelson/ffmt
rev: v0.2.1
hooks:
- id: ffmt
To auto-update to the latest version, run:
pre-commit autoupdate
Or use pre-commit.ci to update hooks automatically via pull requests.
Other Fortran Formatters
| Formatter | Language | Status | Notes |
|---|---|---|---|
| fprettify | Python | Unmaintained (last release 2020) | Free-form only. Fypp support. Requires multiple passes for convergence. |
| findent | C | Active | Indentation and fixed/free conversion. No whitespace normalization. |
| Codee Formatter | Proprietary | Active | Commercial. Tree-sitter based. Fixed and free-form. |
| LFortran fmt | Rust | In development | AST-based. Part of the LFortran compiler project. |
| Fortitude | Rust | Active | Linter with auto-fix, not a full formatter. Tree-sitter based. |
| f90-mode | Emacs Lisp | Active | Emacs built-in. Indentation only. |
License
MIT
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 Distributions
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 ffmt-0.2.7.tar.gz.
File metadata
- Download URL: ffmt-0.2.7.tar.gz
- Upload date:
- Size: 67.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
06ccdfa72cdb5b3916bb0723805853ad98a2702db56e27bd487eb8a144e096f6
|
|
| MD5 |
e84894f10404551e84201f3da97f6a20
|
|
| BLAKE2b-256 |
3bc3fcbb5b9aca6c57e5813439a895c60682ea1467a96a7903bb8597200eb18f
|
Provenance
The following attestation bundles were made for ffmt-0.2.7.tar.gz:
Publisher:
release.yml on sbryngelson/ffmt
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
ffmt-0.2.7.tar.gz -
Subject digest:
06ccdfa72cdb5b3916bb0723805853ad98a2702db56e27bd487eb8a144e096f6 - Sigstore transparency entry: 1155381060
- Sigstore integration time:
-
Permalink:
sbryngelson/ffmt@ba832062304428dc12d57469ca04971b7971c3e6 -
Branch / Tag:
refs/tags/v0.2.7 - Owner: https://github.com/sbryngelson
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@ba832062304428dc12d57469ca04971b7971c3e6 -
Trigger Event:
release
-
Statement type:
File details
Details for the file ffmt-0.2.7-py3-none-win_amd64.whl.
File metadata
- Download URL: ffmt-0.2.7-py3-none-win_amd64.whl
- Upload date:
- Size: 1.5 MB
- Tags: Python 3, Windows x86-64
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6abe2c9a0c8350cf49ccf9913d69d70098e894b5cbf7866a2a7303f6c826746a
|
|
| MD5 |
8340ebae6b4122f3bc4fdb6de66f731f
|
|
| BLAKE2b-256 |
817592392cd68cc4542326f7b91ba3e6a7d62827ad1e4c2081e70242843bb5b9
|
Provenance
The following attestation bundles were made for ffmt-0.2.7-py3-none-win_amd64.whl:
Publisher:
release.yml on sbryngelson/ffmt
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
ffmt-0.2.7-py3-none-win_amd64.whl -
Subject digest:
6abe2c9a0c8350cf49ccf9913d69d70098e894b5cbf7866a2a7303f6c826746a - Sigstore transparency entry: 1155381065
- Sigstore integration time:
-
Permalink:
sbryngelson/ffmt@ba832062304428dc12d57469ca04971b7971c3e6 -
Branch / Tag:
refs/tags/v0.2.7 - Owner: https://github.com/sbryngelson
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@ba832062304428dc12d57469ca04971b7971c3e6 -
Trigger Event:
release
-
Statement type:
File details
Details for the file ffmt-0.2.7-py3-none-manylinux_2_34_x86_64.whl.
File metadata
- Download URL: ffmt-0.2.7-py3-none-manylinux_2_34_x86_64.whl
- Upload date:
- Size: 1.8 MB
- Tags: Python 3, manylinux: glibc 2.34+ x86-64
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
7292607bf02eb7ed67bbb449438893238e4c43cbfeb636c21102a37370e555c7
|
|
| MD5 |
e5cbe90181652a5c4698b29b902b21d2
|
|
| BLAKE2b-256 |
e90c3fae253885680bff37de53eb0c364d5da4adabae41e04d7de3c1bfe88df6
|
Provenance
The following attestation bundles were made for ffmt-0.2.7-py3-none-manylinux_2_34_x86_64.whl:
Publisher:
release.yml on sbryngelson/ffmt
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
ffmt-0.2.7-py3-none-manylinux_2_34_x86_64.whl -
Subject digest:
7292607bf02eb7ed67bbb449438893238e4c43cbfeb636c21102a37370e555c7 - Sigstore transparency entry: 1155381070
- Sigstore integration time:
-
Permalink:
sbryngelson/ffmt@ba832062304428dc12d57469ca04971b7971c3e6 -
Branch / Tag:
refs/tags/v0.2.7 - Owner: https://github.com/sbryngelson
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@ba832062304428dc12d57469ca04971b7971c3e6 -
Trigger Event:
release
-
Statement type:
File details
Details for the file ffmt-0.2.7-py3-none-macosx_11_0_arm64.whl.
File metadata
- Download URL: ffmt-0.2.7-py3-none-macosx_11_0_arm64.whl
- Upload date:
- Size: 1.6 MB
- Tags: Python 3, macOS 11.0+ ARM64
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
f1348e69d687c4b070bfcaf47600f039ed290d53fd6a3c8890283ca9648a12b8
|
|
| MD5 |
9d6c2cb6f86ea3f6f6d3ca54ba6b29e7
|
|
| BLAKE2b-256 |
72bd49eeaa50388ab080dc64c4192560ba536c1d60d618f93ec1c055616965af
|
Provenance
The following attestation bundles were made for ffmt-0.2.7-py3-none-macosx_11_0_arm64.whl:
Publisher:
release.yml on sbryngelson/ffmt
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
ffmt-0.2.7-py3-none-macosx_11_0_arm64.whl -
Subject digest:
f1348e69d687c4b070bfcaf47600f039ed290d53fd6a3c8890283ca9648a12b8 - Sigstore transparency entry: 1155381071
- Sigstore integration time:
-
Permalink:
sbryngelson/ffmt@ba832062304428dc12d57469ca04971b7971c3e6 -
Branch / Tag:
refs/tags/v0.2.7 - Owner: https://github.com/sbryngelson
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@ba832062304428dc12d57469ca04971b7971c3e6 -
Trigger Event:
release
-
Statement type: