Python API¶
Beyond the command line, clat is a small library. Import it to format LaTeX
source from your own scripts, editors, or build tools.
from clat import texfmt, load_config
config = load_config() # nearest .clat.toml, or defaults
result = texfmt(open("main.tex").read(), filename="main.tex", config=config)
print(result.text) # the formatted source
for rule, n_hits in result.clangs: # what was auto-fixed
print(f"clang: {rule.name} ({n_hits})")
for rule, fname, line, msg in result.clunks:
print(f"clunk: {fname}:{line}: {msg}")
If you omit config, texfmt uses the built-in defaults (threshold
5, every rule at its default weight). To run with a custom
threshold without a file on disk, build the dict yourself:
By default, texfmt runs fixable rules to a text fixed point, up to 5 sweeps.
Pass max_iter=1 for single-pass behaviour:
Formatting¶
clat.texfmt
¶
Apply formatting rules according to config.
Returns a ClatResult with the formatted text and categorised issues.
Fixable rules are applied repeatedly until a full sweep makes no text
changes, or until max_iter sweeps have run. Detect-only rules are then
evaluated once against the final text.
Source code in src/clat/rules.py
1105 1106 1107 1108 1109 1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178 1179 1180 1181 1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193 1194 1195 1196 1197 1198 1199 1200 1201 | |
clat.ClatResult
dataclass
¶
Result of running clat on a file.
Attributes:
| Name | Type | Description |
|---|---|---|
text |
str — the (possibly modified) source text
|
|
clangs |
list[tuple] — (rule, count) for auto-fixed rules above threshold
|
|
clunks |
list[tuple] — (rule, filename, line, msg) for unfixable issues above threshold
|
|
splats |
list[tuple] — (rule, filename, line, msg) for issues below threshold
|
|
iterations |
int — number of fixable-rule sweeps performed
|
|
converged |
bool — True if a sweep completed with no text changes
|
|
Source code in src/clat/rules.py
Configuration¶
clat.load_config
¶
Load config from .clat.toml or fallback locations.
Returns a dict with 'threshold' (int), 'weights' (dict[str, int]), 'protected_environments' (list[str]), and 'unprotected_rules' (list[str]).
Source code in src/clat/rules.py
clat.save_config
¶
Write config dict back to a .clat.toml file, preserving all settings.
Source code in src/clat/rules.py
clat.generate_default_config
¶
Return a .clat.toml string with all rules and their default weights.
Source code in src/clat/rules.py
The rule registry¶
clat.Rule
dataclass
¶
A single clat rule.
Attributes:
| Name | Type | Description |
|---|---|---|
id |
str — unique key, used in config overrides (e.g. 'labels_inline')
|
|
name |
str — human-readable description
|
|
fn |
callable — fix function f(text) -> text (fixable=True)
|
or warn function f(text, filename) -> [(file, line, msg)] |
weight |
int — default severity 1–10; 0 disables the rule
|
|
fixable |
bool — True if clat can auto-fix this
|
|
order |
int — execution order (lower = earlier); fixes run before warns
|
|
Source code in src/clat/rules.py
clat.RULES
module-attribute
¶
RULES = [Rule(1, 'labels_inline', 'Merge \\label onto the same line as \\section', rule1_labels_inline, weight=8, fixable=True, order=10), Rule(2, 'decorative_comments', 'Strip decorative comment separators (%%===, %%--- etc.)', rule7_strip_decorative_comments, weight=6, fixable=True, order=20), Rule(3, 'heading_spacing', 'Two blank lines before headings, none after', rule5_heading_spacing, weight=7, fixable=True, order=30), Rule(4, 'equation_separators', 'Insert % lines around display-math environments', rule2_equation_separators, weight=7, fixable=True, order=40), Rule(5, 'equation_punctuation', 'Add trailing comma or period to display equations', rule4_equation_punctuation, weight=6, fixable=True, order=50), Rule(6, 'float_indentation', 'Tab-indent content inside figure/table/list environments', rule6_figure_indentation, weight=5, fixable=True, order=60), Rule(7, 'one_sentence_per_line', 'Split sentences onto individual lines', rule3_one_sentence_per_line, weight=8, fixable=True, order=70), Rule(8, 'math_delimiters_inline', 'Replace \\(...\\) with $...$', rule8_math_delimiters_inline, weight=5, fixable=True, order=80), Rule(9, 'math_delimiters_display', 'Replace \\[...\\] with $$...$$', rule9_math_delimiters_display, weight=0, fixable=True, order=85), Rule(10, 'math_delimiters_equation', 'Replace \\[...\\] or $$...$$ with equation environment', rule10_math_delimiters_equation, weight=0, fixable=True, order=35), Rule(11, 'tilde_before_refs', 'Ensure non-breaking space before \\ref, \\cite etc.', rule11_tilde_before_refs, weight=7, fixable=True, order=90), Rule(12, 'number_unit_spacing', 'Normalise number-unit spacing (100\\,kN)', rule12_number_unit_spacing, weight=6, fixable=True, order=100), Rule(13, 'old_font_commands', 'Replace {\\bf text} with \\textbf{text} etc.', rule13_old_font_commands, weight=5, fixable=True, order=110), Rule(14, 'ellipsis', 'Replace ... with \\dots', rule14_ellipsis, weight=4, fixable=True, order=120), Rule(15, 'ordinal_suffixes', 'Convert superscript ordinals to plain text (1st, 2nd)', rule15_ordinal_suffixes, weight=8, fixable=True, order=130), Rule(16, 'table_line_endings', 'Table \\\\ on row line, \\hline/\\toprule on own line', rule16_table_line_endings, weight=7, fixable=True, order=140), Rule(17, 'abbreviation_spacing', 'Force interword space after e.g., i.e., et al.', rule17_abbreviation_spacing, weight=7, fixable=True, order=145), Rule(18, 'long_file', 'Warn if file exceeds 2000 lines', warn_long_file, weight=3, fixable=False, order=200), Rule(19, 'hardcoded_refs', 'Detect "Figure 3" instead of \\cref{...}', warn_hardcoded_refs, weight=6, fixable=False, order=210), Rule(20, 'manual_sizing', 'Detect \\big, \\Big etc. (prefer \\left/\\right)', warn_manual_sizing, weight=3, fixable=False, order=220), Rule(21, 'float_after_heading', 'Detect float placed directly after a heading', warn_float_after_heading, weight=4, fixable=False, order=230)]
Multi-file discovery¶
For multi-file documents, clat.cli.discover_tex_files expands a list of root
files into the full, ordered, de-duplicated set of .tex files reachable
through \input/\include-style commands — the same traversal the -r flag
uses. See Multi-file documents.
clat.cli.discover_tex_files
¶
Return files plus recursively discovered LaTeX inputs/includes.
Roots are visited in the order provided. Dependencies are depth-first, de-duplicated, and resolved relative to the file that references them. Missing .tex dependencies are included in the returned list so the normal formatter path reports them as missing.