waad / media
Add media to your Laravel application in one place
Requires
- php: ^8.1
- laravel/framework: ^9.0|^10.0|^11.0|^12.0|^13.0
- league/flysystem-aws-s3-v3: ^3.0
Requires (Dev)
- laravel/pint: ^1.0
- orchestra/testbench: ^7.0|^8.0|^9.0|^10.0|^11.0
- pestphp/pest-plugin-laravel: ^1.0|^2.0|^3.0|^4.0
README
Media Files Package
A Laravel package for managing media files across multiple storage drivers (local, S3, and any Laravel filesystem disk) with Eloquent model relationships. An alternative to spatie/laravel-medialibrary.
Authors
Requirements
- PHP >= 8.1
- Laravel 9, 10, 11, 12, or 13
Installation
- Install via Composer:
composer require waad/media
- Publish the configuration and migration files:
php artisan vendor:publish --provider="Waad\Media\MediaServiceProvider"
- Edit the config file at
config/media.php - Run the migrations:
php artisan migrate
- (Optional) Create storage symbolic links:
php artisan media:link
Usage
Setup the Trait
Add the HasMedia trait to any Eloquent model:
use Waad\Media\HasMedia; class Post extends Model { use HasMedia; }
Uploading Media
// Upload a single file $media = $post->addMedia($request->file('image'))->upload(); // Upload multiple files $mediaItems = $post->addMedia($request->file('images'))->upload(); // Upload to a specific collection $media = $post->addMedia($request->file('image')) ->collection('avatars') ->upload(); // Upload with custom options $media = $post->addMedia($request->file('image')) ->collection('gallery') ->disk('s3') ->bucket('photos') ->label('Profile Picture') ->index(3) ->upload();
The upload service works with any Laravel filesystem disk — local, S3, or any custom driver. Just set the disk() and it handles the rest.
Media Collections
Register custom collections in your model to define specific media groups with their own storage configuration:
use Waad\Media\HasMedia; class User extends Model { use HasMedia; public function registerCollections(array $attributes = []): array { return [ 'avatar' => [ 'disk' => 's3', 'bucket' => 'avatars', // directory in the bucket if whant save direct on bucket set empty string '' 'label' => 'User Avatar', 'single' => true, // Only keeps one file 's3' => [ 'ttl_temporary_url' => config('media.s3.default_ttl_temporary_url', 5), ], ], 'gallery' => [ 'disk' => 'public', 'bucket' => 'photos', 'label' => 'Photo Gallery', 'single' => false, // Allows multiple files ], ]; } }
When uploading to a registered collection, its disk, bucket, and label are applied automatically:
$user->addMedia($file)->collection('avatars')->upload();
Retrieving Media
// Get all media $allMedia = $post->getMedia(); // Get media from a specific collection $avatars = $post->getCollection('avatars'); // Get collection urls $urls = $post->getCollectionUrls('avatars'); $urls = $post->getCollectionGroupUrls(); // get urls for multiple collections $urls = $post->getCollectionGroupUrls(only: ['avatar', 'gallery']); // get urls for multiple collections $urls = $post->getCollectionGroupUrls(except: ['avatar', 'gallery']); // get urls for multiple collections // Get all collections by group $collections = $post->getCollectionGroups(); $collections = $post->getCollectionGroups(only: ['avatar', 'gallery']); // only return the collections in the array $collections = $post->getCollectionGroups(except: ['avatar', 'gallery']); // return all collections except the ones in the array // Get collection as array $array = $post->getCollectionArray('avatars'); // Get first or last media $first = $post->getFirstMedia(); $last = $post->getLastMedia(); // Get first or last media by collection $first = $post->getFirstMediaByCollection('gallery'); $last = $post->getLastMediaByCollection('gallery'); // Check if model has media in a collection $post->hasMedia('avatars'); // true / false // Find by ID (supports withTrashed) $media = $post->mediaById($id); $media = $post->mediaById($id, withTrashed: true); // Filter by MIME type $images = $post->mediaByMimeType('image/jpeg'); // Filter by MIME type and collection $images = $post->mediaByMimeTypeByCollection('image/jpeg', 'gallery'); $images = $post->mediaByMimeTypeByCollection('image/jpeg', 'gallery', withTrashed: true); // Filter by approval status $approved = $post->mediaApproved(); $disapproved = $post->mediaApproved(false); // Statistics $totalSize = $post->mediaTotalSize(); $totalCount = $post->mediaTotalCount(); $totalCount = $post->mediaTotalCount(withTrashed: true); // Statistics by collection $size = $post->mediaTotalSizeByCollection('gallery'); $size = $post->mediaTotalSizeByCollection('gallery', withTrashed: true); $count = $post->mediaTotalCountByCollection('gallery'); $count = $post->mediaTotalCountByCollection('gallery', withTrashed: true);
Syncing Media
Use syncMedia() when you want to remove existing rows in a collection and optionally upload replacements. Deletes run before the new upload, and only affect the active collection (the default collection, or the one set with collection() on the chain, or the $collection argument passed to upload()). Media in other collections on the same model is left alone.
Remove specific records, then upload
Pass database IDs of Media rows to soft-delete before new files are stored. IDs that do not belong to the active collection are ignored (no rows removed).
$post->syncMedia($request->file('images'), [$oldMediaId1, $oldMediaId2])->upload(); $post->syncMedia($request->file('image'), [$oldMediaId])->collection('avatars')->upload(); // Illuminate\Support\Collection of IDs is also accepted $post->syncMedia($file, collect([$id1, $id2]))->upload();
Replace everything in the collection
If you omit the second argument (or pass an empty array), existing media in that collection is removed first, then new files are uploaded. This is useful for a full “replace gallery” flow.
$post->syncMedia($request->file('images'))->upload(); $post->syncMedia($request->file('image'))->collection('gallery')->upload();
Additive uploads (no removal)
To upload new files without running the sync removal step, use syncMediaWithoutDettached() or chain setIsWithDettachedSync(false) after syncMedia():
$post->syncMediaWithoutDettached($request->file('extra'))->upload(); $post->syncMedia($request->file('extra'))->setIsWithDettachedSync(false)->upload();
You can still combine with collection(), disk(), and the rest of the fluent API as with addMedia().
Deleting Media
// Delete specific media by ID (soft delete) $post->deleteMedia($mediaId)->delete(); // Delete specific media by model $post->deleteMedia($mediaModel)->delete(); // Delete multiple media by IDs $post->deleteMedia([$id1, $id2, $id3])->delete(); // Delete all media for the model $post->deleteMedia()->delete();
Approving Media
$media->approve(); // Mark as approved $media->disApprove(); // Mark as disapproved // Query approved media globally $approved = \Waad\Media\Media::approved()->get();
File Utilities
The upload service provides disk-aware utility methods that work with any storage driver:
$service = $post->addMedia(null); // Check if a file exists $service->fileExists('upload/photo.jpg'); // Get file size in bytes $service->fileSize('upload/photo.jpg'); // Get file metadata (size, mimetype, last_modified) $service->fileMetadata('upload/photo.jpg'); // Delete a file from disk $service->deleteFile('upload/photo.jpg'); // Generate a temporary URL (S3 and compatible disks) $service->disk('s3')->temporaryUrl('photos/secret.jpg', minutes: 10);
Filament
To use waad/media inside Filament admin panels, install the companion package waad/filament-media. It provides a MediaUpload form component that lines up with HasMedia and registerCollections() (single vs multiple files, disks, reordering via index, and sync behavior).
- Plugin page (docs & requirements): Media Library — Filament
- This package (filament plugin): waad/filament-media
composer require waad/filament-media
See the Filament plugin documentation for setup, MediaUpload::make(...), and table previews with getCollectionUrls().
Configuration
Customize the package in config/media.php:
return [ // Media model class 'model' => \Waad\Media\Media::class, // Database table name 'table_name' => 'media', // Default storage settings 'disk' => env('MEDIA_DISK', 'public'), 'bucket' => env('MEDIA_BUCKET', 'upload'), 'default_collection' => env('MEDIA_DEFAULT_COLLECTION', 'default'), // S3 configuration 's3' => [ 'default_ttl_temporary_url' => env('MEDIA_DEFAULT_S3_TTL_TEMPORARY_URL', 5), ], // Map disk names to public URL prefixes (used by media:link and full_url) 'shortcut' => [ // 'public' => 'storage', ], // Append full_url attribute to the Media model 'enable_full_url' => env('MEDIA_ENABLE_FULL_URL', true), // Auto-prune soft-deleted media after N days (via media:prune) 'prune_media_after_day' => env('MEDIA_PRUNE_MEDIA_AFTER_DAY', 30), // Default approval status for newly uploaded media 'default_approved' => env('MEDIA_DEFAULT_APPROVED', true), // Date format for created_at / updated_at serialization (null = Y-m-d H:i:s) 'format_date' => env('MEDIA_DATE_FORMAT', null), ];
Media Model Attributes
| Attribute | Type | Description |
|---|---|---|
basename |
string | Hashed file name on disk |
filename |
string | Original uploaded file name |
path |
string | Full path in storage |
index |
int | Order index (default 1) |
label |
string | Custom label |
collection |
string | Collection name |
disk |
string | Storage disk (hidden) |
bucket |
string | Storage bucket/folder (hidden) |
mimetype |
string | File MIME type |
filesize |
int | File size in bytes |
approved |
bool | Approval status |
metadata |
json | Additional metadata (e.g. image dimensions) |
full_url |
string | Appended: public URL to the file |
Artisan Commands
| Command | Description |
|---|---|
php artisan media:link |
Create symbolic links from disk roots to public paths based on the shortcut config. Use --force to recreate existing links. |
php artisan media:prune |
Permanently delete soft-deleted media records (and their files) older than prune_media_after_day. |
Features
- Multiple file upload support
- Works with any Laravel filesystem disk (local, S3, etc.)
- Collection management with per-collection disk/bucket configuration
- Soft deletes with
withTrashedsupport - File approval system
- Automatic image metadata extraction (width, height)
- File utilities (exists, size, metadata, delete, temporary URL)
- Automatic file cleanup via
media:prune - Polymorphic media relationships
- File statistics (total size, total count) — global and per-collection
- Customizable date serialization format
syncMedia()with collection-scoped deletes (by ID or full collection replace), plus additive uploads viasyncMediaWithoutDettached()orsetIsWithDettachedSync(false)
Testing
composer install
composer test
License
MIT © Waad Mawlood
