<p align="center">
<br>
<a href="https://github.com/madebyfrmwrk/svgm">
<picture>
<source media="(prefers-color-scheme: dark)" srcset="https://raw.githubusercontent.com/madebyfrmwrk/svgm/main/assets/svgm-dark.svg">
<source media="(prefers-color-scheme: light)" srcset="https://raw.githubusercontent.com/madebyfrmwrk/svgm/main/assets/svgm-light.svg">
<img alt="svgm" src="https://raw.githubusercontent.com/madebyfrmwrk/svgm/main/assets/svgm-light.svg" height="42">
</picture>
</a>
<br>
<br>
</p>
<p align="center">
SVG optimization, rewritten in Rust.
</p>
<p align="center">
<a href="#install">Install</a> ·
<a href="#usage">Usage</a> ·
<a href="#benchmarks">Benchmarks</a> ·
<a href="#how-it-works">How it works</a> ·
<a href="#contributing">Contributing</a>
</p>
<div align="center">
[![MIT licensed][license-badge]][license-url]
[![Crate][crates-badge]][crates-url]
[![Build Status][ci-badge]][ci-url]
</div>
---
## About
SVG files exported from tools like Figma, Illustrator, and Inkscape often include metadata, redundant attributes, unnecessary wrapper structure, and verbose path data.
[SVGO](https://github.com/svg/svgo) has been the standard SVG optimizer for years. SVGM takes a different approach: a native Rust optimizer designed around fixed-point convergence, safe defaults, and a modern CLI. Like [oxlint](https://oxc.rs) for ESLint, SVGM targets the same problem with a different architecture.
### Fixed-point convergence
In some optimizers, additional runs can still reduce output further because later passes create opportunities for earlier ones. SVGM is designed to converge in a single invocation by running optimization passes over the in-memory AST until the document stabilizes.
```
$ svgm icon.svg
icon.svg
13.5 KiB -> 6.6 KiB (51.1% smaller) 0ms 3 passes
```
No re-parsing between iterations. No manual multipass flag. One invocation, fixed-point optimization.
## Install
### From source (requires [Rust](https://rustup.rs))
```bash
cargo install svgm
```
### Build from repo
```bash
git clone https://github.com/madebyfrmwrk/svgm.git
cd svgm
cargo build --release
# Binary at ./target/release/svgm
```
## Usage
```bash
svgm icon.svg # Optimize in place (overwrites the file)
svgm icon.svg -o icon.min.svg # Write to a different file
svgm icon.svg --stdout # Print to stdout instead of overwriting
svgm icon.svg --dry-run # Preview size reduction without writing
svgm icons/*.svg # Optimize multiple files in place
svgm -r ./icons # Recursively optimize all SVGs in a directory
svgm -r ./icons -o ./icons-min # Recursive with output directory
svgm icon.svg --quiet # Suppress all output except errors
```
When piped (e.g. `svgm icon.svg | gzip`), output goes to stdout automatically.
### Presets
```bash
svgm icon.svg --preset safe # Removal + normalization only (17 passes)
svgm icon.svg --preset balanced # Full pass set (24 passes, default)
svgm icon.svg --preset aggressive # Full pass set, lower precision
svgm icon.svg --precision 2 # Override numeric precision on any preset
```
### Config file
Create an `svgm.config.toml` in your project root:
```toml
preset = "balanced"
precision = 3
[passes]
removeDesc = true # opt-in: strip <desc> and <title>
convertShapeToPath = false # opt-out: disable a specific pass
```
SVGM auto-discovers the config by walking up from the input file's directory. Use `--config path` to specify explicitly, or `--no-config` to skip. See [`svgm.config.example.toml`](svgm.config.example.toml) for a starter template.
## Benchmarks
100 real-world SVG logos (Figma, Illustrator, Inkscape, svgrepo exports). 902.7 KiB total original size.
| **Compression** | 14.9% | 18.2% |
| **Median time** | 110ms | 291ms |
| **Speed** | **2.6x faster** | baseline |
| **Invocations to converge** | **1** | 1-3 |
Full benchmark details at [svgm.dev/docs/benchmarks](https://svgm.dev/docs/benchmarks).
SVGM ships as a single native binary. The ~3 point compression gap is actively being closed.
## How it works
### Architecture
```
parse optimize serialize
SVG string ---------> AST tree ---------> AST tree -----------> SVG string
xmlparser fixed-point minified
loop output
```
1. **Parse** — `xmlparser` tokenizes the SVG into an arena-based AST with parent pointers
2. **Optimize** — Run all passes in a loop until no pass reports a change (max 10 iterations)
3. **Serialize** — Write the AST back as a minified SVG string
Passes operate directly on the in-memory AST, avoiding repeated serialize/parse cycles between iterations.
### Optimization passes
**Removal** — strip dead weight
- Comments, doctypes, XML processing instructions
- Editor metadata (Inkscape, Illustrator, Sketch, Figma)
- Empty containers, empty attributes, empty text elements
- Unused namespace declarations
- Attributes matching SVG spec defaults (`opacity="1"`, `stroke="none"`, etc.)
**Normalization** — tighten values
- Collapse whitespace in attributes
- Round numeric values, strip trailing zeros and default `px` units
- Shorten colors: `rgb(255,0,0)` -> `red`, `#aabbcc` -> `#abc`
**Structural** — simplify the tree
- Collapse useless `<g>` wrappers (no-attribute groups, single-child groups)
- Reference safety: groups with `clip-path`, `mask`, or `filter` are never collapsed
**Transform** — simplify and apply transforms
- Merge consecutive transforms into a single equivalent (`translate(10,20) translate(5,5)` -> `translate(15,25)`)
- Remove identity transforms (`scale(1)`, `translate(0,0)`, `rotate(0)`)
- Apply pure translates directly to element coordinates and path data
- Push transforms from single-child groups to child, enabling group collapse
**Geometry** — compress path data
- Shape-to-path conversion (rect, circle, ellipse, line, polyline, polygon → shorter `<path>`)
- Path merging (adjacent non-overlapping paths with identical attributes, with geometric intersection safety)
- Absolute-to-relative coordinate conversion where shorter
- `L` to `H`/`V` shortcut commands
- `C` to `S` and `Q` to `T` shorthand curves (reflected control points)
- Degenerate curve to line simplification (collinear control points)
- Redundant command removal (zero-length lines)
- Strip leading zeros (`.5` instead of `0.5`)
- Implicit command repetition
- Minimal separator insertion
**IDs** — clean up references
- Remove unused `id` attributes
- Shorten referenced IDs to minimal unique names
### Safety
SVGM is conservative by default:
- `<desc>` and `<title>` are **preserved** (accessibility semantics)
- `<symbol>` and `<defs>` with `id` attributes are **never removed** (may be referenced)
- Animation elements (`<animate>`, `<animateTransform>`, etc.) are **fully preserved**
- `<foreignObject>` content is **never touched**
- `fill="black"` on `<svg>` is **kept** (inherited by children)
## Rust API
```rust
use svgm_core::optimize;
let result = optimize("<svg>...</svg>").unwrap();
println!("{}", result.data); // optimized SVG string
println!("{}", result.iterations); // convergence iterations
```
With preset/precision control:
```rust
use svgm_core::{optimize_with_config, Config, Preset};
let config = Config {
preset: Preset::Safe,
precision: Some(2),
..Config::default()
};
let result = optimize_with_config("<svg>...</svg>", &config).unwrap();
```
## Project structure
```
svgm/
├── crates/
│ ├── svgm-core/ # Parser, AST, optimizer, serializer, passes
│ └── svgm-cli/ # CLI binary (clap + indicatif)
├── LICENSE-MIT
└── LICENSE-APACHE
```
## Roadmap
- [x] Transform merging, application, and push-down (all 3 phases)
- [x] Shape-to-path conversion (rect, circle, ellipse → shorter `<path>`)
- [x] Path merging (adjacent paths with identical attributes)
- [x] ID cleanup (remove unused, shorten used)
- [x] CSS `<style>` inlining and minification
- [x] Recursive directory processing (`-r`)
- [x] Safety presets and config file support
- [x] WASM build for browser usage
- [ ] Node.js bindings via napi-rs
## Contributing
SVGM is early, but already usable. Contributions and real-world SVG edge cases are especially helpful.
```bash
git clone https://github.com/madebyfrmwrk/svgm.git
cd svgm
cargo test --workspace
cargo clippy --workspace # must pass clean
```
If you find an SVG that SVGM corrupts or handles worse than expected, please [open an issue](https://github.com/madebyfrmwrk/svgm/issues) with the SVG attached.
## License
Dual-licensed under [MIT](LICENSE-MIT) and [Apache 2.0](LICENSE-APACHE).
[license-badge]: https://img.shields.io/badge/license-MIT%2FApache--2.0-blue.svg
[license-url]: https://github.com/madebyfrmwrk/svgm/blob/main/LICENSE-MIT
[crates-badge]: https://img.shields.io/crates/v/svgm.svg
[crates-url]: https://crates.io/crates/svgm
[ci-badge]: https://github.com/madebyfrmwrk/svgm/actions/workflows/ci.yml/badge.svg
[ci-url]: https://github.com/madebyfrmwrk/svgm/actions