akankov / html-min
HTML compressor and minifier for PHP 8.3+
Fund package maintenance!
Requires
- php: 8.3.* || 8.4.* || 8.5.*
- ext-dom: *
- ext-libxml: *
- ext-mbstring: *
- psr/http-factory: ^1.0
- psr/http-message: ^1.1 || ^2.0
- psr/http-server-middleware: ^1.0
- psr/log: ^3.0
Requires (Dev)
- friendsofphp/php-cs-fixer: ^3.65
- infection/infection: ^0.33
- nyholm/psr7: ^1.8
- phan/phan: ^6.0
- phpstan/phpstan: ^2.1
- phpunit/phpunit: ^12.0
- rector/rector: ^2.0
Suggests
- matthiasmullie/minify: Aggressive inline CSS/JS minification. Wire via HtmlMin::setInlineCssMinifier() / setInlineJsMinifier().
This package is auto-updated.
Last update: 2026-06-12 08:40:06 UTC
README
html-min
A fast HTML5 compressor and minifier for PHP. Strips redundant whitespace, comments, optional tags, and default attributes, then sorts what's left so your gzip layer has less work to do.
Built on native \DOMDocument — no third-party DOM dependencies.
Requirements
- PHP 8.3, 8.4, or 8.5
- ext-dom, ext-libxml, ext-mbstring
Installation
composer require akankov/html-min
Usage
use Akankov\HtmlMin\HtmlMin; $html = <<<HTML <html> <body> <ul style=""> <li style="display: inline;" class="foo">One</li> <li class="foo" style="display: inline;">Two</li> </ul> </body> </html> HTML; echo (new HtmlMin())->minify($html); // <html><body><ul><li class=foo style="display: inline;">One<li class=foo style="display: inline;">Two</ul>
Wrap any block in <nocompress>…</nocompress> to keep its whitespace intact.
Configuration
Every option is a chainable setter. All defaults are shown — the example below reproduces the default configuration.
$htmlMin = (new HtmlMin()) // Core ->doOptimizeViaHtmlDomParser(true) // run the DOM-based pass (required for most of the flags below) ->doRemoveComments(true) // drop HTML comments (conditional comments are preserved) ->doSumUpWhitespace(true) // collapse runs of whitespace in text nodes ->doRemoveWhitespaceAroundTags(false)// aggressive: also trim whitespace adjacent to block tags ->doRemoveSpacesBetweenTags(false) // aggressive: remove whitespace-only text nodes between elements // Attribute optimization ->doOptimizeAttributes(true) ->doSortHtmlAttributes(true) // canonical attribute order → better gzip ->doSortCssClassNames(true) // canonical class order → better gzip ->doRemoveOmittedQuotes(true) // class="foo" → class=foo when safe ->doRemoveOmittedHtmlTags(true) // <p>x</p> → <p>x where the closing tag is optional ->doRemoveOmittedHtmlStartTags(false)// aggressive: also omit the <html>/<head>/<body> START tags ->doRemoveEmptyAttributes(true) ->doRemoveValueFromEmptyInput(true) ->doRemoveDefaultAttributes(false) // opt-in: drop defaults like form method=get // URL attribute trimming ->doRemoveHttpPrefixFromAttributes(false) ->doRemoveHttpsPrefixFromAttributes(false) ->doKeepHttpAndHttpsPrefixOnExternalAttributes(false) ->doMakeSameDomainsLinksRelative([]) // e.g. ['example.com'] → strip host from same-site links // Deprecated attribute cleanup ->doRemoveDeprecatedAnchorName(true) ->doRemoveDeprecatedScriptCharsetAttribute(true) ->doRemoveDeprecatedTypeFromScriptTag(true) ->doRemoveDeprecatedTypeFromStylesheetLink(true) ->doRemoveDeprecatedTypeFromStyleAndLinkTag(true) ->doRemoveDefaultMediaTypeFromStyleAndLinkTag(true) ->doRemoveDefaultTypeFromButton(false) // Inline CSS / JS minification (opt-in, off by default) ->doMinifyInlineCss(false) // minify the contents of inline <style> blocks ->doMinifyInlineJs(false); // minify the contents of inline <script> blocks echo $htmlMin->minify($html);
Each setter returns $this, so you can configure and call minify() in one chain.
Presets
MinifierOptions ships three named starting points so you don't have to
reason about all 27 flags at once:
use Akankov\HtmlMin\Config\MinifierOptions; use Akankov\HtmlMin\HtmlMin; $balanced = new HtmlMin(MinifierOptions::defaults()); // same as new HtmlMin() $smallest = new HtmlMin(MinifierOptions::aggressive()); $shapeStable = new HtmlMin(MinifierOptions::conservative());
defaults()— the balanced configuration shown above. Safe for almost any HTML; output is spec-equivalent to the input.aggressive()— maximum byte savings within what the HTML5 spec allows. Adds block-tag whitespace trimming, whitespace-only text-node removal,<html>/<head>/<body>start-tag omission, spec-default attribute removal, and inline CSS/JS minification. Two trade-offs to know: whitespace between inline elements is rendering-significant, so markup that relies on</span> <span>spacing can render tighter; and start-tag omission can reduce an effectively-empty document — a bare html/head/body skeleton — to an empty string, which is correct per spec but surprising the first time. URL scheme stripping stays off even here because protocol-relative URLs change meaning outside an http(s) context.conservative()— shape-preserving: only collapses whitespace runs, strips comments, and drops spec-redundant deprecated attributes. Keeps optional end tags, attribute quotes, attribute/class order, and empty attributes. Use it when output is diffed against input, post-processed by shape-sensitive tools, or styled via selectors like[data-x=""].
Presets are plain constructors — start from one and override per-flag with the fluent setters if you need a variation.
Inline CSS and JS minification
By default the contents of <style> and <script> blocks round-trip untouched.
Enable the two opt-in toggles to minify them:
$htmlMin = (new HtmlMin()) ->doMinifyInlineCss(true) ->doMinifyInlineJs(true); echo $htmlMin->minify($html);
The bundled minifiers are zero-dependency and conservative:
- CSS — strips
/* … */comments and collapses whitespace; the contents of strings andurl(…)are preserved. - JS — removes comments and collapses horizontal whitespace while preserving newlines (so Automatic Semicolon Insertion is unaffected), strings, regex literals, and template literals. Identifiers are never renamed.
Scripts that are not JavaScript are left alone automatically: a <script>
whose type is, for example, application/ld+json or text/x-template, and any
<script src="…">, passes through unminified.
Using a different minifier
For aggressive minification (identifier renaming, dead-code removal), plug in a
third-party tool with setInlineCssMinifier() / setInlineJsMinifier(). Each
takes any callable(string): string; pass null to restore the bundled default.
use MatthiasMullie\Minify; // composer require matthiasmullie/minify $htmlMin = (new HtmlMin()) ->doMinifyInlineCss(true) ->doMinifyInlineJs(true) ->setInlineCssMinifier(static fn (string $css): string => (new Minify\CSS($css))->minify()) ->setInlineJsMinifier(static fn (string $js): string => (new Minify\JS($js))->minify()); echo $htmlMin->minify($html);
If a bundled minifier throws, the original source is kept and a warning is sent
to the PSR-3 logger (when one is set via setLogger()), so a minifier bug can
never corrupt the page. User-supplied callables let their exceptions propagate.
Command line
The package ships a small CLI at vendor/bin/html-min — handy for one-off
minification, build pipelines, and inspecting what a config change does:
vendor/bin/html-min page.html # minify a file to stdout vendor/bin/html-min page.html --output=page.min.html cat page.html | vendor/bin/html-min # stdin → stdout vendor/bin/html-min page.html --minify-inline-css --minify-inline-js
Exit codes: 0 success, 1 I/O failure (unreadable input / unwritable
--output), 2 invalid argument. --help prints the full usage text.
make phar builds a self-contained dist/html-min.phar for use outside a
Composer project.
Extending
To run your own pass over every element during minification, implement
Akankov\HtmlMin\Contract\DomObserver and register it:
use Akankov\HtmlMin\Contract\DomObserver; use Akankov\HtmlMin\Contract\HtmlMinInterface; use Akankov\HtmlMin\HtmlMin; final class StripDataTestIds implements DomObserver { public function domElementBeforeMinification(\DOMElement $element, HtmlMinInterface $htmlMin): void { } public function domElementAfterMinification(\DOMElement $element, HtmlMinInterface $htmlMin): void { if ($element->hasAttribute('data-testid')) { $element->removeAttribute('data-testid'); } } } $htmlMin = new HtmlMin(); $htmlMin->attachObserverToTheDomLoop(new StripDataTestIds()); echo $htmlMin->minify($html);
Observer practicalities worth knowing before you ship one:
- Phases.
attachObserverToTheDomLoop()takes an optionalObserverPhase(Before,After, or the defaultBoth):domElementBeforeMinification()runs in the pre-pass, before attribute optimization;domElementAfterMinification()in the post-pass. Register for only the phase you need — each hook fires for every element on everyminify()call, so heavy work multiplies across the document. - Ordering. Observers run in registration order within a phase, and the
built-in
OptimizeAttributesobserver is registered first (in theAfterphase) — your after-phase observers see already-optimized attributes. There is no priority system. - Branch on the live config. The second argument is the owning
HtmlMinInterface; itsisDo*()getters reflect the actual flags, so an observer can follow e.g.isDoRemoveHttpPrefixFromAttributes()instead of duplicating configuration. - Exceptions propagate. The DOM walk does not wrap observer calls — a
throwing observer aborts the whole
minify()call. Catch inside the observer if a page must never break on observer bugs. - Mutating the tree is the point, but removing the element you were just handed (or re-parenting its ancestors) mid-walk can skip nodes — prefer attribute and text mutations, and removal of descendants.
Benchmarks
Measured against voku/html-min, wyrihaximus/html-compress, zaininnari/html-minifier, and abordage/html-min on a corpus of real-world HTML pages.
| adapter | median ms/op | geomean ms/op | parse failures | avg gzipped ratio |
|---|---|---|---|---|
| akankov/html-min | 2.4 | 2.4 | 0 / 15 | 90.7% |
| akankov/html-min (inline) | 2.5 | 3.0 | 0 / 15 | 87.6% |
| voku/html-min | 3.6 | 4.1 | 0 / 15 | 90.7% |
| wyrihaximus/html-compress | 7.0 | 8.8 | 0 / 15 | 87.0% |
| zaininnari/html-minifier | 10.8 | 9.3 | 0 / 15 | 94.8% |
| abordage/html-min † | 0.2 | 0.2 | 0 / 15 | 90.2% |
The table above is regenerated by make bench from the latest run.
See latest.md for the per-fixture detail (speed, peak memory,
gzipped compression ratio, methodology, and non-claims). Reproduce with
make bench-install && make bench (requires Docker).
Development
composer install make md-check # markdown formatting (Docker) vendor/bin/phpunit # tests vendor/bin/phpstan analyse # static analysis (level max) vendor/bin/php-cs-fixer fix # code style
CI runs the full matrix (PHP 8.3 / 8.4 / 8.5) on every push and pull request.
License
MIT — see LICENSE.
Originally authored by Lars Moelleken; maintained in this fork by Alex Kankov.