HTML Sanitization
Sanitize untrusted HTML to prevent XSS attacks. Based on bluemonday.
Sanitization works by parsing HTML and filtering it through a whitelist policy. Elements and attributes not explicitly allowed are removed. The output is always well-formed HTML.
Loading
local html = require("html")
Preset Policies
Three built-in policies for common use cases:
| Policy | Use Case | Allows |
|---|---|---|
new_policy |
Custom sanitization | Nothing (build from scratch) |
ugc_policy |
User comments, forums | Common formatting (p, b, i, a, lists, etc.) |
strict_policy |
Plain text extraction | Nothing (strips all HTML) |
Empty Policy
Creates a policy that allows nothing. Use this to build a custom whitelist from scratch.
local policy, err = html.sanitize.new_policy()
policy:allow_elements("p", "strong", "em")
policy:allow_attrs("class"):globally()
local clean = policy:sanitize(user_input)
Returns: Policy, error
User Content Policy
Pre-configured for user-generated content. Allows common formatting elements.
local policy = html.sanitize.ugc_policy()
local safe = policy:sanitize('<p>Hello <strong>world</strong></p>')
-- '<p>Hello <strong>world</strong></p>'
local xss = policy:sanitize('<p>Hello <script>alert("xss")</script></p>')
-- '<p>Hello </p>'
Returns: Policy, error
Strict Policy
Strips all HTML, returns plain text only.
local policy = html.sanitize.strict_policy()
local text = policy:sanitize('<p>Hello <b>world</b>!</p>')
-- 'Hello world!'
Returns: Policy, error
Element Control
Allow Elements
Whitelist specific HTML elements.
local policy = html.sanitize.new_policy()
policy:allow_elements("p", "strong", "em", "br")
policy:allow_elements("h1", "h2", "h3")
policy:allow_elements("a", "img")
local result = policy:sanitize('<p>Hello <strong>world</strong></p>')
-- '<p>Hello <strong>world</strong></p>'
| Parameter | Type | Description |
|---|---|---|
... |
string | Element tag names |
Returns: Policy
Attribute Control
Allow Attributes
Start attribute permission. Chain with on_elements() or globally().
policy:allow_attrs("href"):on_elements("a")
policy:allow_attrs("src", "alt"):on_elements("img")
policy:allow_attrs("class", "id"):globally()
| Parameter | Type | Description |
|---|---|---|
... |
string | Attribute names |
Returns: AttrBuilder
On Specific Elements
Allow attributes only on specific elements.
policy:allow_elements("a", "img")
policy:allow_attrs("href", "target"):on_elements("a")
policy:allow_attrs("src", "alt", "width", "height"):on_elements("img")
| Parameter | Type | Description |
|---|---|---|
... |
string | Element tag names |
Returns: Policy
On All Elements
Allow attributes globally on any permitted element.
policy:allow_attrs("class"):globally()
policy:allow_attrs("id"):globally()
Returns: Policy
With Pattern Matching
Validate attribute values against regex pattern.
-- Only allow hex colors in style
local builder, err = policy:allow_attrs("style"):matching("^color:#[0-9a-fA-F]{6}$")
if err then
return nil, err
end
builder:on_elements("span")
policy:sanitize('<span style="color:#ff0000">Red</span>')
-- '<span style="color:#ff0000">Red</span>'
policy:sanitize('<span style="background:red">Bad</span>')
-- '<span>Bad</span>'
| Parameter | Type | Description |
|---|---|---|
pattern |
string | Regex pattern |
Returns: AttrBuilder, error
URL Security
Standard URLs
Enable URL handling with security defaults.
policy:allow_elements("a")
policy:allow_attrs("href"):on_elements("a")
policy:allow_standard_urls()
Returns: Policy
URL Schemes
Restrict which URL schemes are allowed.
policy:allow_url_schemes("https", "mailto")
policy:sanitize('<a href="https://example.com">OK</a>')
-- '<a href="https://example.com">OK</a>'
policy:sanitize('<a href="javascript:alert(1)">XSS</a>')
-- '<a>XSS</a>'
| Parameter | Type | Description |
|---|---|---|
... |
string | Allowed schemes |
Returns: Policy
Relative URLs
Allow or disallow relative URLs.
policy:allow_relative_urls(true)
policy:sanitize('<a href="/page">Link</a>')
-- '<a href="/page">Link</a>'
| Parameter | Type | Description |
|---|---|---|
allow |
boolean | Allow relative URLs |
Returns: Policy
Nofollow Links
Add rel="nofollow" to all links. Prevents SEO spam.
policy:allow_attrs("href", "rel"):on_elements("a")
policy:require_nofollow_on_links(true)
policy:sanitize('<a href="https://example.com">Link</a>')
-- '<a href="https://example.com" rel="nofollow">Link</a>'
| Parameter | Type | Description |
|---|---|---|
require |
boolean | Add nofollow |
Returns: Policy
Noreferrer Links
Add rel="noreferrer" to all links. Prevents referrer leakage.
policy:require_noreferrer_on_links(true)
| Parameter | Type | Description |
|---|---|---|
require |
boolean | Add noreferrer |
Returns: Policy
External Links in New Tab
Add target="_blank" to fully qualified URLs.
policy:allow_attrs("href", "target"):on_elements("a")
policy:add_target_blank_to_fully_qualified_links(true)
policy:sanitize('<a href="https://example.com">Link</a>')
-- '<a href="https://example.com" target="_blank">Link</a>'
| Parameter | Type | Description |
|---|---|---|
add |
boolean | Add target blank |
Returns: Policy
Convenience Methods
Allow Images
Permit <img> with standard attributes.
policy:allow_images()
policy:sanitize('<img src="photo.jpg" alt="Photo">')
-- '<img src="photo.jpg" alt="Photo">'
Returns: Policy
Allow Data URI Images
Permit base64 embedded images.
policy:allow_elements("img")
policy:allow_attrs("src"):on_elements("img")
policy:allow_data_uri_images()
policy:sanitize('<img src="...">')
-- '<img src="...">'
Returns: Policy
Allow Lists
Permit list elements: ul, ol, li, dl, dt, dd.
policy:allow_lists()
policy:sanitize('<ul><li>Item 1</li><li>Item 2</li></ul>')
-- '<ul><li>Item 1</li><li>Item 2</li></ul>'
Returns: Policy
Allow Tables
Permit table elements: table, thead, tbody, tfoot, tr, td, th, caption.
policy:allow_tables()
policy:sanitize('<table><tr><td>Cell</td></tr></table>')
-- '<table><tr><td>Cell</td></tr></table>'
Returns: Policy
Allow Standard Attributes
Permit common attributes: id, class, title, dir, lang.
policy:allow_elements("p")
policy:allow_standard_attributes()
policy:sanitize('<p id="intro" class="text" title="Introduction">Hello</p>')
-- '<p id="intro" class="text" title="Introduction">Hello</p>'
Returns: Policy
Sanitize
Apply policy to HTML string.
local policy = html.sanitize.ugc_policy()
policy:require_nofollow_on_links(true)
local dirty = '<p>Hello</p><script>alert("xss")</script>'
local clean = policy:sanitize(dirty)
-- '<p>Hello</p>'
| Parameter | Type | Description |
|---|---|---|
html |
string | HTML to sanitize |
Returns: string
Errors
| Condition | Kind | Retryable |
|---|---|---|
| Invalid regex pattern | errors.INVALID |
no |
See Error Handling for working with errors.