softcreatr / php-mime-detector
Modern, extensible MIME type detector for PHP
Fund package maintenance!
softcreatr
Installs: 306 264
Dependents: 4
Suggesters: 0
Security: 0
Stars: 25
Watchers: 3
Forks: 4
Open Issues: 0
pkg:composer/softcreatr/php-mime-detector
Requires
- php: >=8.1.0
- ext-ctype: *
Requires (Dev)
- phpunit/phpunit: >=10.0
- roave/security-advisories: dev-latest
- squizlabs/php_codesniffer: ^3.10
Suggests
- ext-zip: *
README
A modern, extensible MIME type detector for PHP that analyses the actual bytes of a file instead of trusting its extension. The detector ships with a modular pipeline of signature matchers and a bidirectional repository of MIME type ↔ extension mappings so you can integrate it into security-sensitive workflows.
Features
- Real file inspection – identifies file formats by signature instead of relying on filenames. You can find a list of supported file formats in the Wiki.
- Composable architecture – category-specific detectors can be swapped in or extended without touching the core.
- Rich lookup helpers – translate between MIME types and extensions in both directions and enumerate the supported catalogue.
- No external dependencies – runs on any PHP 8.1+ installation without and can take
requiring additional, external extensions, or packages, but takes advantage
of
ZipArchivewhen it is available.
Installation
Install the package via Composer:
composer require softcreatr/php-mime-detector
Quick start
Detecting the MIME type and the preferred extension for a file is as simple as instantiating the façade and calling its helpers:
<?php use SoftCreatR\MimeDetector\MimeDetector; use SoftCreatR\MimeDetector\MimeDetectorException; require 'vendor/autoload.php'; try { $detector = new MimeDetector(__DIR__ . '/example.png'); echo $detector->getMimeType(); // image/png echo $detector->getFileExtension(); // png echo $detector->getFileHash(); // crc32 hash of the file contents } catch (MimeDetectorException $exception) { // React to unreadable files or unsupported formats. echo $exception->getMessage(); }
Resolving MIME types and extensions
The façade exposes several lookup helpers that do not require a file scan. They operate on the shared repository of known mappings:
$detector = new MimeDetector(__DIR__ . '/example.png'); // Retrieve the canonical extension for a MIME type. $extension = $detector->getExtensionForMimeType('image/jpeg'); // "jpg" // List every MIME type that corresponds to the given extension. $mimeTypes = $detector->getMimeTypesForExtension('heic'); // Fetch the complete map as [mimeType => list of extensions]. $catalogue = $detector->listAllMimeTypes();
Need a data URI? The detector will encode the configured file for you:
$dataUri = $detector->getBase64DataURI(); // data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAA...
Optional ZipArchive support
The detector is fully functional without PHP's ZipArchive extension; all ZIP
signatures are recognised by scanning the first 4 KiB of the file for well-known
markers such as mimetype, [Content_Types].xml, or classes.dex. When the
extension is present, the ZipSignatureDetector opens the archive and inspects
its entries directly. This deeper look allows the detector to resolve format
families like OOXML (.docx, .pptx, .xlsx), APK/JAR/XPI bundles, and other
ZIP-based containers even when their identifying files live deeper inside the
archive than the cached bytes.
If the extension is missing, the detector simply falls back to its heuristic
path and ultimately reports a generic application/zip match whenever a more
specific signature cannot be derived. Unit tests that require ZipArchive are
skipped automatically when the class is not available, so no additional setup is
needed to run the suite.
Extending the detector
Custom formats can be added without modifying the library itself. Choose the approach that fits your needs best.
Registering detectors via extensions
When you only need to extend the default pipeline, extensions are the quickest
way to plug in extra detectors. The snippet below is a complete, copy & paste ready
bootstrap that you can drop into a service provider, bootstrap.php, or any
other file that runs before you instantiate the façade:
<?php declare(strict_types=1); use App\MimeDetector\CustomContainerDetector; use SoftCreatR\MimeDetector\MimeDetector; use SoftCreatR\MimeDetector\MimeDetectorException; use SoftCreatR\MimeDetector\MimeTypeDetector; use SoftCreatR\MimeDetector\MimeTypeRepository; require __DIR__ . '/vendor/autoload.php'; // 1) Teach the repository about your MIME type ↔ extension mapping. $repository = MimeTypeRepository::createDefault(); $repository->register('custom', 'application/x-custom'); // 2) Register one or more detectors as an extension. Higher priorities run first. MimeTypeDetector::extend( 'custom-container', static function (): array { return [ new CustomContainerDetector(), // More detectors can be returned from the same extension when needed. ]; }, priority: 150, ); // 3) Resolve files like usual – the default pipeline now includes your extension. try { $detector = new MimeDetector(__DIR__ . '/file.cust', $repository); echo $detector->getMimeType(); // application/x-custom echo $detector->getFileExtension(); // custom } catch (MimeDetectorException $exception) { echo $exception->getMessage(); }
Extensions can be forgotten at runtime (MimeTypeDetector::forgetExtension('custom-container'))
or reset entirely (MimeTypeDetector::flushExtensions()). Returning multiple
detectors from an extension lets you register related matchers in one go while still
benefiting from the priority-based ordering.
Building a custom pipeline
For more advanced scenarios you can compose a bespoke pipeline. Follow these steps to teach the detector about a new signature and MIME mapping.
1. Implement a signature detector
Create a class that implements
SoftCreatR\MimeDetector\Contract\FileSignatureDetectorInterface. The detector
receives the DetectionContext, which gives access to the file buffer and lets
you return a MimeTypeMatch when the signature is recognised.
<?php namespace App\MimeDetector; use SoftCreatR\MimeDetector\Attribute\DetectorCategory; use SoftCreatR\MimeDetector\Contract\FileSignatureDetectorInterface; use SoftCreatR\MimeDetector\Detection\DetectionContext; use SoftCreatR\MimeDetector\Detection\MimeTypeMatch; #[DetectorCategory('custom')] final class CustomContainerDetector implements FileSignatureDetectorInterface { public function detect(DetectionContext $context): ?MimeTypeMatch { $buffer = $context->buffer(); if ($buffer->checkForBytes([0x43, 0x55, 0x53, 0x54])) { // "CUST" return new MimeTypeMatch('custom', 'application/x-custom'); } return null; } }
2. Register MIME mappings
Extend the repository so your MIME type resolves to the expected extension(s):
use SoftCreatR\MimeDetector\MimeTypeRepository; $repository = MimeTypeRepository::createDefault(); $repository->register('custom', 'application/x-custom');
3. Compose (or customise) the detector pipeline
Most projects do not need to rebuild the pipeline manually. Once an extension is registered it is automatically merged with the default signature detectors in priority order:
use SoftCreatR\MimeDetector\MimeDetector; use SoftCreatR\MimeDetector\MimeTypeDetector; MimeTypeDetector::extend( 'custom-container', new CustomContainerDetector(), priority: 50, // run before the bundled detectors ); $detector = new MimeDetector(__DIR__ . '/file.cust', $repository); $match = $detector->getMimeType(); // application/x-custom-container
If you do need full control you can still provide a bespoke pipeline. Simply prepend your detector to the default ones so it executes before the fallback signatures:
use SoftCreatR\MimeDetector\Detection\DetectorPipeline; use SoftCreatR\MimeDetector\Detector\ArchiveSignatureDetector; use SoftCreatR\MimeDetector\Detector\DocumentSignatureDetector; use SoftCreatR\MimeDetector\Detector\ExecutableSignatureDetector; use SoftCreatR\MimeDetector\Detector\FontSignatureDetector; use SoftCreatR\MimeDetector\Detector\ImageSignatureDetector; use SoftCreatR\MimeDetector\Detector\MediaSignatureDetector; use SoftCreatR\MimeDetector\Detector\MiscSignatureDetector; use SoftCreatR\MimeDetector\Detector\XmlSignatureDetector; use SoftCreatR\MimeDetector\Detector\ZipSignatureDetector; use SoftCreatR\MimeDetector\MimeDetector; $pipeline = DetectorPipeline::create( new CustomContainerDetector(), new ImageSignatureDetector(), new ZipSignatureDetector(), new ArchiveSignatureDetector(), new MediaSignatureDetector(), new DocumentSignatureDetector(), new FontSignatureDetector(), new ExecutableSignatureDetector(), new MiscSignatureDetector(), new XmlSignatureDetector(), ); $detector = new MimeDetector(__DIR__ . '/file.cust', $repository, $pipeline);
From this point the new MIME type behaves exactly like the built-in ones – it can be detected from files, resolved by MIME type, and listed in the catalogue.
Testing
Fixture files for the test suite are stored in a Git submodule. After cloning this repository run:
git submodule update --init --recursive
composer install
composer test
Contributing
We welcome pull requests! Please review CONTRIBUTING for the coding standards and workflow. When adding new detections, include at least one fixture so behaviour can be verified automatically.
License
Released under the ISC License.