overtrue / laravel-versionable
Make Laravel model versionable.
Fund package maintenance!
overtrue
Installs: 228 752
Dependents: 3
Suggesters: 0
Security: 0
Stars: 572
Watchers: 10
Forks: 48
Open Issues: 11
pkg:composer/overtrue/laravel-versionable
Requires
- php: ^8.1
- doctrine/dbal: ^3.8|^4.0
- jfcherng/php-diff: ^6.11
- laravel/framework: ^9.0|^10.0|^11.0|^12.0
Requires (Dev)
- brainmaestro/composer-git-hooks: dev-master
- laravel/pint: ^1.5
- mockery/mockery: ^1.4
- orchestra/testbench: ^8.21|^10.0
- phpunit/phpunit: ^10.0|^11.5.3|^12.0
- 5.x-dev
- 5.5.0
- 5.4.0
- 5.3.3
- 5.3.2
- 5.3.1
- 5.3.0
- 5.2.7
- 5.2.6
- 5.2.5
- 5.2.4
- 5.2.3
- 5.2.2
- 5.2.1
- 5.2.0
- 5.1.4
- 5.1.3
- 5.1.2
- 5.1.1
- 5.1.0
- 5.0.0
- 5.0.0-beta.1
- 4.x-dev
- 4.6.5
- 4.6.4
- 4.6.3
- 4.6.0
- 4.5.0
- 4.4.0
- 4.3.0
- 4.2.1
- 4.2.0
- 4.1.0
- 4.0.2
- 4.0.1
- 4.0.0
- 3.x-dev
- 3.1.4
- 3.1.3
- 3.1.2
- 3.1.1
- 3.1.0
- 3.0.1
- 3.0.0
- 2.x-dev
- 2.6.1
- 2.6.0
- 2.5.0
- 2.4.0
- 2.3.0
- 2.2.1
- 2.2.0
- 2.1.0
- 2.0.0
- 1.x-dev
- 1.1.5
- 1.1.4
- 1.1.3
- 1.1.2
- 1.1.1
- 1.1.0
- 1.0.1
- 1.0.0
This package is auto-updated.
Last update: 2025-10-21 16:36:06 UTC
README
It's a minimalist way to make your model support version history, and it's very simple to revert to the specified version.
Requirement
- PHP >= 8.1.0
- laravel/framework >= 9.0
Features
- Keep the specified number of versions.
- Whitelist and blacklist for versionable attributes.
- Easily revert to the specified version.
- Record only changed attributes.
- Easy to customize.
Installing
composer require overtrue/laravel-versionable -vvv
First, publish the config file and migrations:
php artisan vendor:publish --provider="Overtrue\LaravelVersionable\ServiceProvider"
Then run this command to create a database migration:
php artisan migrate
Usage
Add Overtrue\LaravelVersionable\Versionable
trait to the model and set versionable attributes:
use Overtrue\LaravelVersionable\Versionable; class Post extends Model { use Versionable; /** * Versionable attributes * * @var array */ protected $versionable = ['title', 'content']; // Or use a blacklist //protected $dontVersionable = ['created_at', 'updated_at']; <...> }
Versions will be created on the vensionable model saved.
$post = Post::create(['title' => 'version1', 'content' => 'version1 content']); $post->update(['title' => 'version2']);
Get versions
$post->versions; // all versions $post->latestVersion; // latest version // or $post->lastVersion; $post->versions->first(); // first version // or $post->firstVersion; $post->versionAt('2022-10-06 12:00:00'); // get version from a specific time // or $post->versionAt(\Carbon\Carbon::create(2022, 10, 6, 12));
Revert
Revert a model instance to the specified version:
$post->getVersion(3)->revert(); // or $post->revertToVersion(3);
Revert without saving
$version = $post->versions()->first(); $post = $version->revertWithoutSaving();
Remove versions
// soft delete $post->removeVersion($versionId = 1); $post->removeVersions($versionIds = [1, 2, 3]); $post->removeAllVersions(); // force delete $post->forceRemoveVersion($versionId = 1); $post->forceRemoveVersions($versionIds = [1, 2, 3]); $post->forceRemoveAllVersions();
Restore deleted version by id
$post->restoreTrashedVersion($id);
Temporarily disable versioning
// create Post::withoutVersion(function () use (&$post) { Post::create(['title' => 'version1', 'content' => 'version1 content']); }); // update Post::withoutVersion(function () use ($post) { $post->update(['title' => 'updated']); });
Custom Version Store strategy
You can set the following different version policies through property protected $versionStrategy
:
Overtrue\LaravelVersionable\VersionStrategy::DIFF
- Version content will only contain changed attributes (default strategy).Overtrue\LaravelVersionable\VersionStrategy::SNAPSHOT
- Version content will contain all versionable attribute values.
Show diff between the two versions
$diff = $post->getVersion(1)->diff($post->getVersion(2));
$diff
is a object Overtrue\LaravelVersionable\Diff
, it based
on jfcherng/php-diff.
You can render the diff to many formats, and all formats result will be like follows:
[ $attribute1 => $diffOfAttribute1, $attribute2 => $diffOfAttribute2, ... $attributeN => $diffOfAttributeN, ]
toArray()
$diff->toArray(); // [ "name" => [ "old" => "John", "new" => "Doe", ], "age" => [ "old" => 25, "new" => 26, ], ]
Other formats
toArray(array $differOptions = [], array $renderOptions = [], bool $stripTags = false): array toText(array $differOptions = [], array $renderOptions = [], bool $stripTags = false): array toJsonText(array $differOptions = [], array $renderOptions = [], bool $stripTags = false): array toContextText(array $differOptions = [], array $renderOptions = [], bool $stripTags = false): array toHtml(array $differOptions = [], array $renderOptions = [], bool $stripTags = false): array toInlineHtml(array $differOptions = [], array $renderOptions = [], bool $stripTags = false): array toJsonHtml(array $differOptions = [], array $renderOptions = [], bool $stripTags = false): array toSideBySideHtml(array $differOptions = [], array $renderOptions = [], bool $stripTags = false): array
Note
$differOptions
and$renderOptions
are optional, you can set them following the README of jfcherng/php-diff.$stripTags
allows you to remove HTML tags from the Diff, helpful when you don't want to show tags.
Using custom version model
You can define $versionModel
in a model, that used this trait to change the model(table) for versions
Note
Model MUST extend class
\Overtrue\LaravelVersionable\Version
;
<?php class PostVersion extends \Overtrue\LaravelVersionable\Version { // }
Update the model attribute $versionModel
:
<?php namespace App\Models; use Illuminate\Database\Eloquent\Model; use Overtrue\LaravelVersionable\Versionable; class Post extends Model { use Versionable; public string $versionModel = PostVersion::class; }
Intergrations
- mansoorkhan96/filament-versionable Effortlessly manage revisions of your Eloquent models in Filament.
❤️ Sponsor me
如果你喜欢我的项目并想支持它,点击这里 ❤️
Project supported by JetBrains
Many thanks to Jetbrains for kindly providing a license for me to work on this and other open-source projects.
Contributing
You can contribute in one of three ways:
- File bug reports using the issue tracker.
- Answer questions or fix bugs on the issue tracker.
- Contribute new features or update the wiki.
The code contribution process is not very formal. You just need to make sure that you follow the PSR-0, PSR-1, and PSR-2 coding guidelines. Any new code contributions must be accompanied by unit tests where applicable.
License
MIT