karimms / model-pulse
Add collaboration, messaging, and activity tracking to any Eloquent model.
v1.0.0
2026-03-22 22:01 UTC
Requires
- php: ^8.3
- illuminate/container: ^12.0|^13.0
- illuminate/contracts: ^12.0|^13.0
- illuminate/database: ^12.0|^13.0
- spatie/eloquent-sortable: ^5.0
Requires (Dev)
- larastan/larastan: ^3.9
- laravel/pint: ^1.0
- pestphp/pest: ^3.0|^4.0
- pestphp/pest-plugin-laravel: ^3.0|^4.1
This package is auto-updated.
Last update: 2026-04-22 22:24:41 UTC
README
Add collaboration, messaging, attachments, activity logs, and domain events to any Eloquent model.
Features
- Follow/unfollow support for any model (
Followable) - Thread-like messaging APIs (
Messagable) - File attachment helpers (
Attachable) - Automatic model activity logging (
HasLogActivity) - Typed Laravel events for messaging and activity operations
Requirements
- PHP 8.3+
- Laravel components:
illuminate/container^12|^13illuminate/contracts^12|^13illuminate/database^12|^13
Installation
composer require karimms/model-pulse
Run package migrations:
php artisan migrate --path=vendor/karimms/model-pulse/database/migrations
Quick Start
Add traits to your Eloquent model:
<?php namespace App\Models; use Illuminate\Database\Eloquent\Model; use Karim\ModelPulse\Traits\Attachable; use Karim\ModelPulse\Traits\Followable; use Karim\ModelPulse\Traits\HasLogActivity; use Karim\ModelPulse\Traits\Messagable; class Project extends Model { use Followable; use Messagable; use Attachable; use HasLogActivity; public function getModelTitle(): string { return $this->name ?? 'Project'; } public function getLogAttributeLabels(): array { return [ 'name' => 'Name', 'status' => 'Status', ]; } }
Usage
Collaboration (Followers)
$project->addFollower($user); $project->isFollowedBy($user); $project->removeFollower($user);
Messaging
$message = $project->addMessage([ 'type' => 'note', 'subject' => 'Kickoff', 'body' => 'Project started', ]); $reply = $project->replyToMessage($message, [ 'type' => 'note', 'subject' => 'Re: Kickoff', 'body' => 'Acknowledged', ]); $project->markAsRead(); $project->pinMessage($message); $project->unpinMessage($message); $project->removeMessage($message->id);
Attachments
$attachments = $project->addAttachments([ 'uploads/spec.pdf', 'uploads/mockup.png', ]); $project->getImageAttachments(); $project->getDocumentAttachments(); $project->removeAttachment($attachments->first()->id);
Custom Storage Disk
Set a global attachment disk in configuration:
// config/model-pulse.php return [ 'attachments' => [ 'disk' => 's3', ], ];
Optional per-model override:
public function getModelPulseAttachmentDisk(): ?string { return 's3'; }
Resolution order:
- Model override (
getModelPulseAttachmentDisk) model-pulse.attachments.diskfilesystems.default
Activity Logging
HasLogActivity automatically records model lifecycle activity and stores audit-style entries through the package message model.
You can also trigger logging manually:
$project->logModelActivity('updated');
Events
The package dispatches typed Laravel events after successful messaging/activity operations.
Messaging Events
Karim\ModelPulse\Events\MessageCreatedKarim\ModelPulse\Events\MessageRepliedKarim\ModelPulse\Events\MessageRemovedKarim\ModelPulse\Events\MessagesMarkedReadKarim\ModelPulse\Events\MessagePinnedKarim\ModelPulse\Events\MessageUnpinned
Activity Event
Karim\ModelPulse\Events\ActivityLogged
Listening to Events
Example listener registration in your EventServiceProvider:
use App\Listeners\HandleMessageCreated; use Karim\ModelPulse\Events\MessageCreated; protected $listen = [ MessageCreated::class => [ HandleMessageCreated::class, ], ];
Notes
- Events are dispatched only when operations succeed.
- Failed or no-op operations do not dispatch success events.