disarm¶
Unicode canonicalization and TR39 visual confusable analysis — building blocks for text-security pipelines (homoglyph/bidi/zalgo/invisible-character handling), plus standards-based phonetic transliteration. One pure-Rust core, with bindings for Python, Ruby, and more.
Documentation | API Reference | PyPI
Get started in your language: Python · Rust · Ruby
Demo¶
Why disarm¶
The text-cleaning libraries already in most pipelines — ftfy, unidecode, anyascii — were built for encoding repair and ASCII conversion. They map confusables phonetically (Cyrillic р → Latin r), which does not reverse a homoglyph substitution.
disarm's normalize_confusables() / strip_obfuscation() implement visual confusable mapping per Unicode TR39 (Cyrillic р → Latin p) — not to be confused with transliterate(), which romanizes phonetically (р → r) like the tools above. Measured over a broad sample of the TR39 confusable space — the 1,314 single-codepoint sources whose skeleton is a single Latin letter (of TR39's 6,565) — this visual mapping recovers XMR = 0.634 (strip_obfuscation) to 0.682 (full pipeline), neutralizing ~95% of sources at the per-source level, where phonetic transliterators stay at or below 0.19:
| Tool class | Mapping | Homoglyph XMR (broad TR39 sample) |
|---|---|---|
| NFKC (compatibility folding) | — | 0.103 |
unidecode, anyascii, cyrtranslit, uroman |
phonetic | ≤ 0.187 |
disarm (strip_obfuscation / normalize_confusables) |
visual (TR39) | 0.634 / 0.682 (95% CI 0.603–0.664 / 0.652–0.710) |
| TR39 skeleton — shares the attack table (oracle ceiling) | visual | 1.000 |
On the original curated cut (18 hand-curated Cyrillic look-alike pairs) disarm reproduces XMR = 1.000 exactly — a labeled sanity check, not the headline. ftfy was statistically equivalent to no preprocessing; unidecode degraded accuracy on invisible-character attacks. Details: Adversarial-Text Defense (paper "Fire Extinguishers Full of Gasoline"; XMR metric: Zenodo 10.5281/zenodo.20618323).
Scope. disarm is a defense-in-depth layer, not a complete control. It canonicalizes the confusables it bundles (TR39) and strips the format characters it enumerates; it does not promise to stop any attack class, and the confusable space is far larger than any table. See the Threat Model for what is and isn't in scope.
Not an output sanitizer. disarm normalizes input; it does not make text safe to emit into HTML, JS, URLs, SQL, or shells. It performs no escaping and does not strip
<,>,&—<script>alert(1)</script>passes through unchanged, and NFKC normalization can even surface ASCII metacharacters from fullwidth lookalikes (<script>→<script>). disarm is not an XSS or injection defense and never replaces one: encode at the output sink (framework auto-escaping, DOMPurify, parameterized queries). Run disarm before those, as the Unicode layer they don't cover.
Which function do I want?¶
The most common confusion is reaching for transliterate() to defend against homoglyphs. It does the opposite mapping. Two distinct operations, two different tables:
| If you want to… | Use | Mapping | Example |
|---|---|---|---|
| Defend against homoglyph / look-alike spoofing | normalize_confusables(), strip_obfuscation() |
visual (TR39) | Cyrillic р → Latin p |
| Romanize text to readable ASCII | transliterate() |
phonetic (BGN/PCGN, ISO 9, GOST) | Cyrillic р → Latin r; Київ → Kyiv (uk profile) |
| Flag spoofed hostnames / IDNs | is_suspicious_hostname() |
analysis (no rewrite) | аpple.com → suspicious |
transliterate() is a romanizer, not a security control: it maps by sound/standard, so it will turn a Cyrillic р into r and leave the spoof readable. For homoglyph defense, always use the visual (TR39) functions in row 1.
from disarm import strip_obfuscation, normalize_confusables, is_suspicious_hostname
# Fold Cyrillic look-alikes to their Latin prototypes (TR39 visual mapping)
assert strip_obfuscation("рroduсt") == 'product'
assert strip_obfuscation("pаypаl 🔥🔥") == 'paypal fire fire'
assert normalize_confusables("раypal") == 'paypal'
# IDN / hostname spoofing check (flags the bad; a False result is not a safety guarantee)
suspicious, analysis = is_suspicious_hostname("аpple.com") # leading Cyrillic а
# suspicious is True; analysis.has_confusables and analysis.mixed_script flag why
Installation¶
pip install disarm
Install and import use the same name, disarm:
import disarm
Requires Python 3.10+. Wheels are available for Linux, macOS, and Windows.
Use from Rust¶
disarm is also a standalone Rust crate. The default build is pure Rust — no
Python, no pyo3, no libpython — so it drops into any Rust project as an
ordinary dependency:
cargo add disarm
The public surface is the disarm::api
module plus the error types (Error, ErrorKind, ErrorMode). The
DisarmStr extension
trait gives the same operations method syntax on any string:
use disarm::{api, DisarmStr};
use disarm::api::{Transliterate, Scheme, OnUnknown, TargetScript};
fn main() {
// TR39 confusable folding (Cyrillic look-alikes → Latin)
assert_eq!(api::normalize_confusables("раypal", TargetScript::Latin), "paypal");
// …or via the extension trait:
assert_eq!("раypal".normalize_confusables(TargetScript::Latin), "paypal");
// Transliteration to ASCII — the one-liner, or the builder for full control
assert_eq!(api::transliterate("Москва"), "Moskva");
let s = Transliterate::new()
.scheme(Scheme::StrictIso9)
.on_unknown(OnUnknown::Replace("?".into()))
.run("Москва");
assert!(s.is_ascii());
// Canonicalization primitives (borrow on the no-op path via Cow)
assert_eq!(api::strip_accents("café"), "cafe");
assert_eq!(api::fold_case("fi"), "fi");
assert_eq!(api::slugify("Héllo Wörld", &api::SlugConfig::default()), "hello-world");
// IDN / hostname spoofing check (returns a HostnameAnalysis struct)
let analysis = api::is_suspicious_hostname("раypal.com");
assert!(analysis.suspicious);
}
Fallible operations (sanitize_filename, decode_to_utf8, strip_log_injection,
the key/clean presets) return Result<_, disarm::Error>; inspect
Error::kind() for a
stable ErrorKind.
The extension-module Cargo feature (which pulls in pyo3) is used only to
build the Python wheel — Rust consumers never enable it. See the Rust API &
semver policy and the full reference on
docs.rs/disarm.
Logging (opt-in, off by default)¶
disarm can emit diagnostic records through the binding-neutral
log facade behind the log Cargo feature. It is
off by default — the shipped artifact has no logging code in the hot path
unless you turn it on — and records carry only metadata (lengths, language,
mode, flags, counts, durations, error codes), never the input or output
text. Pick a sink in your application (env_logger, tracing-subscriber, …):
disarm = { version = "0.10", features = ["log"] }
env_logger::init(); // your sink, your level filter
// Core transforms (transliterate, the registration/seal config calls, …) then
// emit redacted records — lengths, flags, counts, duration — but never the text.
A library must not set log's release_max_level_* (those unify across the
whole dependency graph) — that ceiling is the application's call.
Features¶
- Confusable & homoglyph analysis (TR39): visual confusable mapping, bidi-control / zalgo / zero-width / invisible-character stripping, and the
strip_obfuscationpipeline (defense-in-depth — see the Threat Model) - Canonicalization pipelines:
security_clean,normalize_user_input,catalog_key,search_key,sort_key,display_clean,ml_normalizefor common workflows - LLM / RAG pipelines: guardrail matching (
llm_guardrail) and ingestion (rag_ingest) profiles — deterministic deobfuscation and ASCII-index normalisation for LLM stacks - Hostname / IDN analysis: mixed-script and confusable detection for domains
- Standards-based transliteration: best-in-class Latin / Cyrillic / Greek with ISO 9-style ASCII (
strict_iso9), GOST R 7.0.34, and BGN/PCGN, plus reverse transliteration (Russian, Ukrainian, Greek) - Text normalization: NFC/NFD/NFKC/NFKD, full Unicode case folding (1,557 CaseFolding.txt mappings via PHF), whitespace collapse
- Slugification & filename sanitization: URL-safe slugs (python-slugify compatible) and cross-platform safe filenames with path-traversal handling
- Grapheme clusters: correct user-perceived character counting, splitting, and truncation
- Encoding detection: auto-detect and decode byte sequences to UTF-8 (chardetng)
- Broad transliteration coverage for CJK, Indic, and other scripts — a context-free unidecode-compatible drop-in (best-effort; see caveats)
All text processing is implemented in Rust with O(1) PHF lookups and exposed to Python via PyO3.
Quick start¶
Defense & canonicalization¶
from disarm import (
is_confusable, normalize_confusables, strip_obfuscation,
security_clean, normalize_user_input,
)
assert is_confusable("аpple") == True
assert normalize_confusables("раypal") == 'paypal'
# Maximum deobfuscation: homoglyphs, zalgo, invisible chars, bidi, emoji → clean text
assert strip_obfuscation("рroduсt") == 'product'
# Pipelines
assert security_clean("ℝ𝕖𝕒𝕝 𝕥𝕖𝕩𝕥") == 'Real text'
assert normalize_user_input("pаypal") == 'paypal'
Transliteration (standards-based core)¶
Romanization, not homoglyph defense.
transliterate()maps phonetically (Cyrillicр→r), not by TR39 visual confusability (р→p). It will not reverse a look-alike spoof — for that usenormalize_confusables()/strip_obfuscation().
from disarm import transliterate, slugify
assert transliterate("café") == 'cafe'
assert transliterate("Москва") == 'Moskva'
assert transliterate("Αθήνα") == 'Athina'
# Named standards (Latin / Cyrillic / Greek)
assert transliterate("Юрий", strict_iso9=True) == 'Jurij'
assert transliterate("Москва", gost7034=True) == 'Moskva'
# Language profiles (sparse overrides on top of the default table)
assert transliterate("Ärger", lang="de") == 'Aerger'
assert transliterate("Київ", lang="uk") == 'Kyiv'
# Auto-detect language from script
assert transliterate("Москва", lang="auto") == 'Moskva'
# Reverse transliteration (Latin → native script): Russian, Ukrainian, Greek
assert transliterate("Moskva", target="ru") == 'Москва'
assert transliterate("Athina", target="el") == 'Αθηνα'
# Slugs & filenames
assert slugify("café au lait") == 'cafe-au-lait'
Compatibility coverage (CJK and other scripts)¶
# Context-free, character-by-character — best-effort, unidecode-parity (see caveats below)
assert transliterate("北京市") == 'bei jing shi'
assert transliterate("서울") == 'seo ul'
assert transliterate("ひらがな") == 'hiragana'
Coverage tiers¶
disarm transliterates a very wide range of scripts, but the quality guarantee differs by tier. Lead with the core; treat the rest as compatibility coverage.
| Tier | Scripts | Policy | Standard |
|---|---|---|---|
| Core (best-in-class) | Latin, Cyrillic, Greek | Standards-based romanization + reverse | BGN/PCGN (default), ISO 9-style ASCII (strict_iso9), GOST R 7.0.34 (gost7034) |
| Compatibility (best-effort) | CJK (Chinese / Japanese / Korean), Arabic, Hebrew, Devanagari & 9 other Indic scripts, Thai, Lao | Context-free, character-by-character — same approach as Unidecode/AnyAscii | Unihan kMandarin, Revised Romanization, Hepburn, UNGEGN/IAST-derived, RTGS-derived |
| Best-effort | Georgian, Armenian, and a long tail of additional scripts | Context-free coverage so input is never silently dropped | see Language support |
Compatibility-tier transliteration is context-free and character-by-character — no linguistic analysis, polyphony handling, or phonological rules. For CJK/Arabic/Indic this is fundamentally lossy and no better than Unidecode; it exists so disarm is a complete drop-in, not because it is best-in-class there. See limitations.md for trade-offs and the full per-script policy table.
Context-aware abjad (Arabic, Persian, Hebrew): an optional dictionary-backed mode (
transliterate(text, context=True)) restores vowels for more readable output. It is a best-effort readability aid, not a romanization standard. See Abjad scripts.
Precompiled pipelines¶
from disarm import security_clean, ml_normalize, catalog_key, normalize_user_input, strip_obfuscation
# Security: NFKC → confusables → strip bidi → collapse whitespace → path-safety
assert security_clean("ℝ𝕖𝕒𝕝 𝕥𝕖𝕩𝕥") == 'Real text'
# ML/NLP: NFKC → emoji→text → transliterate → strip accents → fold case
assert ml_normalize("Café ☕ Ünïcödé") == 'cafe hot beverage unicode'
# Library catalog: NFKC → transliterate → confusables → strip accents → fold case
assert catalog_key("Москва", lang="ru") == 'moskva'
assert catalog_key("ΩMEGA café") == 'omega cafe'
# Web input: NFKC → strip bidi → strip zero-width → strip control → strip zalgo → confusables → collapse → path-safety
assert normalize_user_input("pаypal") == 'paypal'
# Maximum deobfuscation: homoglyphs, zalgo, invisible chars → clean text
assert strip_obfuscation("рroduсt") == 'product'
assert strip_obfuscation("pаypаl 🔥🔥") == 'paypal fire fire'
# Note: does NOT transliterate — chain with transliterate() if needed
Text builder¶
from disarm import Text
result = (
Text("Ünïcödé Café ☕")
.normalize(form="NFKC")
.demojize()
.transliterate()
.strip_accents()
.fold_case()
.value
)
assert result == 'unicode cafe hot beverage'
Package structure¶
The API is organized into domain-specific namespaces. All functions are also available at the top level for convenience.
| Namespace | Purpose | Key functions |
|---|---|---|
disarm.security |
Defense & safety analysis | normalize_confusables, is_confusable, is_mixed_script, is_suspicious_hostname, strip_bidi, security_clean |
disarm |
Core transforms | transliterate, slugify, strip_obfuscation, Text, TextPipeline |
disarm.normalization |
Unicode normalization | normalize, strip_accents, fold_case, collapse_whitespace |
disarm.files |
Filename handling | sanitize_filename |
disarm.codec |
Byte decoding | decode_to_utf8, detect_encoding |
# Namespace imports
from disarm.security import is_confusable, security_clean
from disarm.codec import decode_to_utf8
from disarm.normalization import fold_case
# Top-level imports also work
from disarm import is_confusable, security_clean, decode_to_utf8, fold_case
Language profiles¶
Built-in language profiles span the core and compatibility tiers, with scholarly ASCII Cyrillic support (strict_iso9; ISO 9-style digraphs, not the diacritic standard). Profiles apply sparse overrides on top of the default table (e.g. German maps ü → ue instead of the default u).
from disarm import list_langs
# 83 built-in language profiles — see Language support for the full registry
assert len(list_langs()) == 83
assert {"de", "uk", "ja-kunrei", "vai"} <= set(list_langs())
See Language support for the full registry, per-script policies, and tier classification.
Performance¶
disarm is compiled Rust with O(1) compile-time perfect hash tables — no regex, no per-character Python iteration, no runtime data loading. Speed is a supporting benefit, not the headline; correctness and defense come first.
Performance is measured in two regimes, because they stress different things. Long text (documents, batch pipelines) is dominated by per-character cost; short strings (per-record processing — names, titles, slugs, one field at a time) are dominated by fixed per-call overhead. disarm is fast in both, and quotes them separately so neither number overstates the other.
Long text — document-scale throughput:
| Operation | Throughput | vs. legacy |
|---|---|---|
| Transliterate (Latin) | ~450M chars/sec | ~38× faster than Unidecode |
| Transliterate (Cyrillic) | ~106M chars/sec | ~15× faster than Unidecode |
| Slugify | ~712K slugs/sec | ~10–24× faster than python-slugify |
| Batch transliterate (100 strings) | ~2.8× faster than loop | — |
Short strings — per-call, ~70–85 character inputs:
| Input | vs. Unidecode |
|---|---|
| Latin | ~17× |
| Mixed scripts | ~14× |
| Cyrillic / Greek | ~13× |
A transliterate() call crosses the Python→Rust boundary exactly once, and
already-ASCII input returns the original str object in roughly 65 ns with
zero allocation. disarm also wins all four cells of Unidecode's own
benchmark — a faithful replication of the
original, re-measured continuously in CI — from ~1.3× on Unidecode's strongest
case (ASCII passthrough) to ~25×. That bar is worth clearing precisely because
Unidecode has carried this workload for two decades; it remains the reference
point this library measures itself against.
Throughput figures are from a commodity 4‑vCPU x86‑64 Linux runner (min‑of‑N
perf_counter); per-call figures are interleaved ratios against pinned
comparator versions on CI runners, median-of-7, bucketed by CPU
microarchitecture, and measured in the fresh-string regime — every timed
call receives a newly constructed str object, as production traffic does,
rather than re-running one cached object (which would understate comparators'
real-world parity and overstate ours). All figures are hardware‑dependent and
directional, not guarantees. See performance.md
for full benchmark methodology and results.
Drop-in replacement¶
disarm provides compatibility aliases for painless migration from existing libraries:
from disarm import unidecode, casefold, remove_accents
assert unidecode("café") == 'cafe'
assert casefold("Straße") == 'strasse'
assert remove_accents("café") == 'cafe'
sanitize_filename() also accepts replacement_text and max_len kwargs for pathvalidate compatibility, and is_confusable() accepts greedy for confusable_homoglyphs compatibility. See migration guides for details.
Security note: the
unidecodealias is for coverage compatibility only. For security/defense use it is the wrong tool (phonetic mapping does not reverse homoglyph attacks and can degrade downstream accuracy). Usestrip_obfuscation/normalize_confusablesinstead — see Migration from Unidecode.
Exhaustive testing¶
disarm is exhaustively tested with three layers of machine-verifiable assurance beyond conventional unit and property-based tests:
- Compile-time assertions:
build.rsasserts all transliteration table values are ASCII and entry counts match expectations — if any check fails,cargo buildfails - Exhaustive domain coverage: Every Hangul syllable (11,172), every BMP codepoint (63,488), every CJK ideograph (20,992), and every Indic script block are tested individually — zero sampling gaps
- Stated invariants: Seven stated properties (ASCII passthrough, idempotence, determinism, output bounds, etc.) verified by exhaustive enumeration and Hypothesis
See formal-verification.md for details.
User Guide¶
Core concepts and usage for each feature area.
- Getting Started — install + quickstart for Python · Rust · Ruby
- Adversarial-Text Defense — TR39 visual confusable mapping vs phonetic transliteration, the XMR benchmark, and why it matters
- Transliteration — Unicode → ASCII with language profiles, plus reverse (Latin → native script)
- Slugification — URL-safe slug generation, drop-in python-slugify replacement
- Normalization — NFC / NFD / NFKC / NFKD Unicode normalization
- Confusable Detection — TR39 homoglyph detection and normalization
- Filename Sanitization — Cross-platform safe filenames
- Text Cleaning — Accent stripping, case folding, whitespace collapse
- Grapheme Clusters — User-perceived character counting, splitting, and truncation
- Text Pipeline — Composable, pre-compiled multi-step processing
- Language Support — Built-in profiles, auto-detection, custom profiles
- Abjad Scripts — Context-aware Arabic, Persian, and Hebrew with dictionary-based vowel restoration
- Language Detection — How
lang="auto"works: script identification, character-level discrimination, fail-safe fallbacks
- Policy Templates — Named institutional presets for libraries, web apps, ML, and more
- CLI — Command-line usage, piping, and shell integration
API Reference¶
Complete function signatures, parameters, and return types.
- Overview — API reference index
- Core Transforms —
transliterate,slugify,normalize,sanitize_filename,strip_accents,strip_zalgo,fold_case,collapse_whitespace,demojize,strip_bidi(all acceptstrorlist[str]) - Precompiled Pipelines —
security_clean,ml_normalize,catalog_key,display_clean,search_key,sort_key,normalize_user_input,PRESETS,get_pipeline,list_profiles - Classes —
Text,Slugifier,UniqueSlugifier,TextPipeline, compatibility aliases - Predicates —
detect_scripts,inspect_auto_lang,is_mixed_script,is_confusable,is_ascii,is_normalized,is_zalgo,is_suspicious_hostname - Grapheme Clusters —
grapheme_len,grapheme_split,grapheme_truncate - Encoding Detection —
detect_encoding,decode_to_utf8 - Language Profiles —
list_langs,register_lang,register_replacements - Enums & Types —
Script,NF,EmojiProvider, type aliases, language constants - Exceptions —
DisarmError
Reference¶
- Language Reference — All languages: codes, names, reference texts, and per-language transliteration rule tables
- Provenance — Standards and sources behind every transliteration mapping
Architecture¶
Internal design documentation for contributors and advanced users.
- Transliteration Engine — PHF lookup, language table chain, Indic virama handling
- Data Tables — TSV format, build.rs code generation, compile-time PHF
- Pipeline — TextPipeline internals, execution order, step bitflags
- Emoji Engine — Emoji detection, provider system, pure-Rust path
- Emoji Plugins — EmojiProvider protocol, custom providers
- Security — Confusable detection, hostname validation, bidi stripping
- Performance — Optimization strategies, PHF tables, batch amortization
- Testing & Guarantees — Test philosophy, property-based testing, security invariants, CI matrix
- Exhaustive Testing — Compile-time assertions, exhaustive domain coverage, stated invariants (I1–I7)
- Transliteration Comparison — Character-level diff vs Unidecode and anyascii
Benchmarks¶
- Performance Overview — Benchmark results: throughput and per-call speedups vs Unidecode, python-slugify, and pathvalidate
- Benchmark Suite — How to run benchmarks, Criterion and timeit configurations
Migration Guides¶
Parameter-compatible replacements for existing libraries.
- Migration Overview — Feature comparison matrix
- From Unidecode / text-unidecode — Drop-in
unidecode()alias - From python-slugify / awesome-slugify — Parameter-compatible
slugify() - From confusable_homoglyphs — Script detection and normalization
- From pathvalidate — Filename sanitization
- From anyascii — Language-aware transliteration
Other¶
- Limitations — Known constraints, edge cases, and design trade-offs