restruct/silverstripe-svg-images

SVG Image support for Silverstripe with real manipulation and sanitization

Installs: 3 778

Dependents: 0

Suggesters: 0

Security: 0

Stars: 13

Watchers: 5

Forks: 20

Open Issues: 2

Type:silverstripe-vendormodule

pkg:composer/restruct/silverstripe-svg-images

2.1.0 2026-01-24 11:30 UTC

This package is auto-updated.

Last update: 2026-01-27 13:38:42 UTC


README

This module provides comprehensive SVG support in SilverStripe's asset management system, including:

  • CMS thumbnail/preview support for SVG files in AssetAdmin
  • Real SVG manipulation (resize, crop) that modifies viewBox/dimensions while preserving vectors
  • SVG sanitization on upload to remove potentially dangerous content
  • Dimension parsing from SVG viewBox/width/height attributes
  • Automatic class handling for SVGs uploaded through Image relations

How it works

The module configures SilverStripe to use the SVGImage class for .svg files via class_for_file_extension. This happens automatically for files uploaded through AssetAdmin.

SVG uploads through Image relations

When uploading SVGs through relation fields (has_one, has_many, or many_many to Image), SilverStripe's ORM enforces the relation's class type, ignoring the class_for_file_extension config. This module includes SVGImageExtension which automatically corrects the ClassName to SVGImage after the file is written.

This happens transparently - no configuration needed.

SVG Manipulation

Unlike raster images, SVG manipulation preserves the vector format by modifying viewBox and width/height attributes. The module uses contao/imagine-svg for manipulation.

Supported operations:

  • Fit($width, $height) - Resize to fit within bounds, maintaining aspect ratio
  • FitMax($width, $height) - Same as Fit, but only if image is larger
  • Fill($width, $height) - Crop and resize to fill exact dimensions
  • FillMax($width, $height) - Same as Fill, but only if image is larger
  • Pad($width, $height) - Fit within bounds and add transparent padding to reach exact dimensions
  • ScaleWidth($width) - Scale to specific width, maintaining aspect ratio
  • ScaleHeight($height) - Scale to specific height, maintaining aspect ratio
  • CropRegion($x, $y, $width, $height) - Crop to specific region

Manipulated SVGs are stored as variants (just like raster image variants), so they're cached and only generated once.

To disable manipulation and return original SVGs unchanged (legacy behavior):

Restruct\Silverstripe\SVG\SVGImage:
  enable_svg_manipulation: false

SVG Sanitization

SVG files are automatically sanitized on upload using enshrined/svg-sanitize. This removes potentially dangerous content like:

  • JavaScript/event handlers
  • External references (can be disabled)
  • PHP tags
  • Other XSS vectors

Configuration options:

Restruct\Silverstripe\SVG\SVGImage:
  # Disable sanitization (not recommended)
  sanitize_on_upload: false

  # Keep remote references (disabled by default for security)
  sanitize_remove_remote_references: false

Migrating existing SVG files

If you have existing SVG files in your database that were uploaded before installing this module, enable auto-migration:

Restruct\Silverstripe\SVG\SVGImage:
  auto_migrate_svg_class: true

Then run dev/build. The migration will update the ClassName in File, File_Live, and File_Versions tables (including files with NULL or empty ClassName).

Note: The migration runs via requireDefaultRecords(). If you use dev/build --no-populate, the migration will be skipped. Run a normal dev/build without --no-populate to trigger it.

Usage in templates

<!-- Basic usage -->
<img src="$Image.URL" />

<!-- With manipulation (preserves vector format) -->
<img src="$Image.ScaleWidth(200).URL" />
<img src="$Image.Fill(100, 100).URL" />
<img src="$Image.Fit(300, 200).URL" />

<!-- Responsive example -->
<img src="$Image.ScaleWidth(400).URL" srcset="$Image.ScaleWidth(800).URL 2x" />

Inline SVG

<!-- Add raw SVG inline -->
{$Image.SVG_RAW_Inline}

<!-- Conditional based on file type -->
<% if $Image.IsSVG %>
  {$Image.SVG_RAW_Inline}
<% else %>
  <img src="$Image.ScaleWidth(400).URL" />
<% end_if %>

Inline SVG with color manipulation

If you need to manipulate SVG colors or add CSS classes for inline SVGs, consider stevie-mayhew/silverstripe-svg. You can use it alongside this module by passing the asset path:

{$SVG($Image.Filename).fill('#FF9933').extraClass('my-icon')}

SVG Security

SVGs can expose attack vectors comparable to HTML/JS. This module mitigates risks through automatic sanitization, but you should still:

  • Only accept SVG uploads from trusted users
  • Use <img> tags rather than inline SVG when possible (provides more browser security)
  • Keep the sanitization enabled (default)

For more information on SVG security risks, see OWASP SVG Security Cheat Sheet.

Configuration reference

Restruct\Silverstripe\SVG\SVGImage:
  # Enable real SVG manipulation (resize/crop)
  enable_svg_manipulation: true

  # Sanitize SVGs on upload
  sanitize_on_upload: true

  # Remove remote references during sanitization
  sanitize_remove_remote_references: true

  # Auto-migrate existing SVG files on dev/build
  auto_migrate_svg_class: false

Development Tools

SVG vs PNG Comparison Tool

A visual comparison tool is available at /dev/svg-compare to verify that SVG manipulations behave consistently with PNG manipulations. The tool:

  • Compares all manipulation methods (Fit, Fill, Pad, Scale, etc.) side-by-side
  • Tests both published and draft/protected assets
  • Includes bundled test images or accepts custom image IDs

Clear SVG Variants Task

To clear all generated SVG variant files (useful after upgrading or when manipulation settings change):

# Dry run - shows what would be deleted
vendor/bin/sake tasks:ClearSVGVariantsTask

# Actually delete variants
vendor/bin/sake tasks:ClearSVGVariantsTask --confirm

# Verbose output
vendor/bin/sake tasks:ClearSVGVariantsTask --confirm --verbose

Variants will be regenerated on next request using the current manipulation settings.

Technical Documentation

For implementation details, architecture decisions, and explanations of why things are implemented in specific ways, see TECHNICAL.md.