dcodegroup / activity-log
Simple package which dcode uses to manage activity logs
Installs: 44 345
Dependents: 0
Suggesters: 0
Security: 0
Stars: 4
Watchers: 6
Forks: 0
Open Issues: 1
Language:JavaScript
Requires
- php: ^8.2 || ^8.3 || ^8.4
- coduo/php-to-string: ^3.2
- laravel/framework: ^10.0 || ^11.0 || ^12.0
- nesbot/carbon: ^2.72 || ^3.0
- spatie/laravel-query-builder: ^4.0.4 || ^5.7 || ^6.0
Requires (Dev)
- ergebnis/composer-normalize: ^2.42
- larastan/larastan: ^2.9
- laravel/pint: ^1.13
- orchestra/testbench: ^8.21
- phpstan/phpstan-deprecation-rules: ^1.1
- phpstan/phpstan-phpunit: ^1.3
- spatie/laravel-log-dumper: ^1.5
- dev-main
- 2.0.19
- 2.0.18
- 2.0.17
- 2.0.16
- 2.0.15
- 2.0.14
- 2.0.13
- 2.0.12
- 2.0.11
- 2.0.10
- 2.0.9
- 2.0.8
- 2.0.7
- 2.0.6
- 2.0.5
- 2.0.4
- 2.0.3
- 2.0.2
- 2.0.1
- 2.0.0
- 1.1.8
- 1.1.7
- 1.1.6
- 1.1.5
- 1.1.4
- 1.1.3
- 1.1.2
- 1.1.1
- 1.1.0
- 1.0.25
- 1.0.24
- 1.0.23
- 1.0.22
- 1.0.21
- 1.0.20
- 1.0.19
- 1.0.18
- 1.0.17
- 1.0.16
- 1.0.15
- 1.0.14
- 1.0.13
- 1.0.12
- 1.0.11
- 1.0.10
- 1.0.9
- 1.0.8
- 1.0.7
- 1.0.6
- 1.0.5
- 1.0.4
- 1.0.3
- 1.0.2
- 1.0.1
- 1.0.0
- dev-feature/DCO-3116-add-events-to-activity-log
- dev-feature/add-testing-setup
- dev-feature/laravel-12
- dev-feature/DCO-3114-activity-log-add-uuid-to-the-activity-log-record
- dev-bugfix/send_notification_email
- dev-feature/FLO-1240-next-step-of-the-activity-log-can-we-now-link-this-up-that-if-i-enter-@
- dev-feature/Feature-update-read-me
- dev-fixBug/update-table-activity-log
- dev-feature/test
This package is auto-updated.
Last update: 2025-03-20 02:56:16 UTC
README
The dcodegroup/activity-log
package provides a simple and unified approach to track and record activity / interactions
against your Laravel models (and relations). Capture changes, updates, and user interactions to enhance transparency and
auditing in your application in a centralised and consistent approach.
Update note
Since version 1.1.1
we are no longer need to use observers to listen for changes from the
model. bootActivityLoggable
in ActivityLoggable
trait solved that. Make sure to remove duplicate observers before
updating
Installation
Add the following to your package.json file:
"dependencies": { ... "floating-vue": "^5.2.2", "vue-markdown-render": "^2.1.1", "@dcodegroup/vue-mention": "^0.0.2", }
PHP
You can install the package via composer:
composer require dcodegroup/activity-log
Then run the install command.
php artisan activity-log:install
Run the migrations
php artisan migrate
User Model
Add the following contract to the User
model.
<?php namespace App\Models; use Dcodegroup\ActivityLog\Contracts\HasActivityUser; class User extends Authenticatable implements HasActivityUser { public function getActivityLogUserName(): string { return $this->name; } public function getActivityLogEmail(): string { return $this->email; } public function getActivityLogUser(): array { return [ 'id' => $this->id, 'full_name' => $this->getActivityLogUserName(), 'email' => $this->getActivityLogEmail(), ]; }
JS
Add the following alias to vite.config.js
resolve: { alias: { "@dcodegroup": path.resolve(__dirname, "./vendor/dcodegroup/"),
Add the following js to your index.js
file.
import VActivityLog from "@dcodegroup/activity-log/resources/js/components/VActivityLog.vue"; import ActivityLogList from "@dcodegroup/activity-log/resources/js/components/ActivityLogList.vue"; import ActivityEmail from "@dcodegroup/activity-log/resources/js/components/ActivityEmail.vue"; app.component("VActivityLog", VActivityLog); app.component("ActivityLogList", ActivityLogList); app.component("ActivityEmail", ActivityEmail);
In your app.scss
file add the following
@import "@dcodegroup/activity-log/resources/sass/index.scss"; @import "floating-vue/dist/style.css";
Seem to need this in tailwind.config.js
under spacing:
spacing: { "3xlSpace": "96px", "2xlSpace": "64px", xlSpace: "32px", lgSpace: "24px", mdSpace: "16px", smSpace: "12px", xsSpace: "8px", "2xsSpace": "4px", "3xsSpace": "2px", }
Update the module exports under content:
content: [ ... "./vendor/dcodegroup/**/*.{blade.php,vue,js,ts}", ... ],
Update the vue il8n package to load additional paths
i18n({ // you can also change your langPath here // langPath: 'locales' additionalLangPaths: [ "vendor/dcodegroup/activity-log/lang", // Load translations from this path too! ], }),
Run the npm build (dev/prod)
npm run prod:assets
Configuration
Most of configuration has been set the fair defaults. However you can review the configuration file
at config/activity-log.php
and adjust as needed
<?php use Dcodegroup\ActivityLog\Models\ActivityLog; return [ /* |-------------------------------------------------------------------------- | Middleware |-------------------------------------------------------------------------- | | What middleware should the package apply. | */ 'middleware' => ['web', 'auth'], /* |-------------------------------------------------------------------------- | Routing |-------------------------------------------------------------------------- | | Here you can configure the route paths and route name variables. | | What should the route path for the activity log be | eg 'api/generic/activity-logs' | | What should the route name for the activity log be | eg eg 'api.generic.activity-logs', */ 'route_path' => env('LARAVEL_ACTIVITY_LOG_ROUTE_PATH', 'activity-logs'), 'route_name' => env('LARAVEL_ACTIVITY_LOG_ROUTE_NAME', 'activity-logs'), /* |-------------------------------------------------------------------------- | Model and Binding |-------------------------------------------------------------------------- | | binding - eg 'activity-logs' | model - eg 'ActivityLog' | */ 'binding' => env('LARAVEL_ACTIVITY_LOG_MODEL_BINDING', 'activity-logs'), 'activity_log_model' => ActivityLog::class, /* |-------------------------------------------------------------------------- | Formatting |-------------------------------------------------------------------------- | | Configuration here is for display configuration | */ 'datetime_format' => env('LARAVEL_ACTIVITY_LOG_DATETIME_FORMAT', 'd-m-Y H:ia'), 'date_format' => env('LARAVEL_ACTIVITY_LOG_DATE_FORMAT', 'd.m.Y'), /* |-------------------------------------------------------------------------- | Pagination |-------------------------------------------------------------------------- | | Configuration here is for pagination | */ 'default_filter_pagination' => env('LARAVEL_ACTIVITY_LOG_PAGINATION', 50), /* |-------------------------------------------------------------------------- | User |-------------------------------------------------------------------------- | | Configuration here is for the user model and table | eg 'User' */ 'user_relationship' => env('LARAVEL_ACTIVITY_LOG_USER_RELATIONSHIP', 'user'), 'user_model' => \App\Models\User::class, 'user_table' => env('LARAVEL_ACTIVITY_LOG_USERS_TABLE', 'users'), /* |-------------------------------------------------------------------------- | Communication log |-------------------------------------------------------------------------- | | */ 'communication_log_model' => \Dcodegroup\ActivityLog\Models\CommunicationLog::class, 'communication_log_table' => env('LARAVEL_ACTIVITY_LOG_COMMUNICATION_LOG_TABLE', 'communication_logs'), 'communication_log_relationship' => env('LARAVEL_ACTIVITY_LOG_COMMUNICATION_LOG_RELATIONSHIP', 'communicationLog'), /* |-------------------------------------------------------------------------- | Filter Builder |-------------------------------------------------------------------------- | | Configuration here is for the filter builder | eg 'FilterBuilder class: App\Support\QueryBuilder\Filters\FilterBuilder' */ 'filter_builder_path' => env('LARAVEL_ACTIVITY_LOG_FILTER_BUILDER_PATH', ''), /* |-------------------------------------------------------------------------- | Events |-------------------------------------------------------------------------- | | Configuration here is for the events | eg 'open_modal_event' => 'openModal' */ 'open_modal_event' => env('LARAVEL_ACTIVITY_LOG_EVENT_OPEN_MODEL', 'openModal'), 'reload_event' => env('LARAVEL_ACTIVITY_LOG_EVENT_RELOAD', 'getActivities'), ];
Value | Options | Description |
---|---|---|
middleware | Include a specification of what middleware this package should include. | |
layout_path | The dot notation path to the resource/view that you would like to use for the Activity Log. | |
content_section | The variable in the view that will contain the output of the Activtity Log. |
Usage
The package provides an endpoints which you can use. See the full list by running
php artisan route:list --name=activity-log
They are
[example.com/activity-logs] Which is where you will form index. This is by default protected auth middleware but you can modify in the configuration. This is where you want to link to in your admin and possibly a new window
QueryBuilder Filters
Located in
src\Support\QueryBuilder\Filters\DateRangeFilter.php
src\Support\QueryBuilder\Filters\TermFilter.php
Traits for activity log model
Located in
src\Support\Traits\ActivityLoggable.php
src\Support\Traits\LastModifiedBy.php
ActivityLoggable Trait
The ActivityLoggable
trait provides functionality for logging activities and communication logs related to a model.
Methods
logCreate(): void
: Automatically create activity log every time a new model is created. (support from version 1.1.1)logUpdate(): void
: Automatically create activity log every time when model is updated. (support from version 1.1.1)logDelete(): void
: Automatically create activity log every time when model is deleted. (support from version 1.1.1)getActivityLogEmails(): array
: Get the emails associated with activity logs.activities(): Collection
: Get the collection of activities associated with the model.modelRelation(): Collection
: Get the relationship between the model column with the related table.modelRelation
also has the effect of limiting logging to defined columns instead of logging all changes from the model when you declaregetModelChangesJson(true)
Example of define modelRelation via model using ActivityLoggable
public function modelRelation(): Collection { return collect([ 'account_id' => collect([ // column change in model 'label' => 'Account', // attribute label display in activity log description 'modelClass' => Account::class, // relationship model 'modelKey' => 'name', // columns display instead ]), ...... ])
when declared like this instead of activity log shows like this. account_id: 1 -> 20
The result will return like
this: Account: Alison Cahill -> Annie Pollock
.
getModelChanges(?array $modelChangesJson = null): string
: Get the model changes as a formatted string.getModelChangesJson(bool $allowCustomAttribute = false): array
: Get the model changes as an array of JSON. If$allowCustomAttribute
=true
If we want to limit the storage of fields defined in modelRelation;false
: If we want to storage all model changecreateActivityLog(array $description): ActivityLog
: Create a new activity log.
Example of define activity log via model using ActivityLoggable
// Creating an activity log $activityLog = $model->createActivityLog([ 'type' => \Dcodegroup\ActivityLog\Models\ActivityLog::TYPE_DATA // if type is null default type will be TYPE_DATA, we support 3 other types: TYPE_STATUS, TYPE_COMMENT, TYPE_NOTIFICATION 'title' => 'Updated profile information', 'description' => 'Updated user profile information', // Additional custom fields as needed 'communication_log_id' => '' // required when type = TYPE_NOTIFICATION to link activity log with communication log ]);
If you have a user case where you want the log messages to be logged against another model, Example. You have an Order
model and you want the OrderItem
models to be recorded against the Order
. Then do as below.
with the OrderItem
model add the method targetModel
class OrderItem extends Model { ... public function targetModel(): self|Model { return $this->order; } }
Normally a model with have the field name
title
or label
. This package can work this out in most cases. However if you have a none standard field used to name a model you can use the below method to customise the label for the model.
If no label is found then a ModelLabelNotDefinedException
exception will be thrown.
public function getActivityLogModelLabel(): string { /** * This can be any field or method to return the label but the return must be a string */ return $this->reference; }
You can give any model a custom label by adding the following method to the model. If this is not set then Str::headline will be used on the model.
public function getActivityLogModelLabel(): string { /** * This can be any field or method to return the label but the return must be a string */ return __('order.title'); }
Automatically the package will try and find the key for the model. Typically, this will be a field named name
, title
or label
.
However this may not always be the case and the key may change depending on the the state of a model. Eg type Quote might be quote_number
Order might use sales_order_number
.
If one of the defaults is not found then an ModelKeyNotDefinedException
exception will be thrown.
This should only ever occur in your local environment. If this occurs then implement the follow method in your model.
public function getActivityLogModelKey(): string { return (string) $this->custom_field_name; }
By default this package will log all fields except for created_at
, updated_at
, deleted_at
, password
, and id
.
If you wish to exclude other fields on your model such as third party api tokens. Then implement the following method in your model.
public function getActivityLogModelExcludeFields(): array { return ['xero_api_token', 'stripe_api_token']; }
You can use a custom formatter for fields in your model by using the `activityLogFieldFormatters` method. example. Add the following to the model ```php use Illuminate\Support\Collection; ... public function activityLogFieldFormatters(): Collection { return collect([ 'price' => fn ($value) => Number::currency(($value / 100), 'AUD'), ]); }
price
is the key for the field.
Right hand side should be a closure than can then be used for format the value that will be present.
You can override the default name / label for an entity. Simply create a method named `` that returns a string. Below is the default.
public function activityLogEntityName(): string { return Arr::join(Str::ucsplit(class_basename($this)), ' ').' (id:'.$this->id.')'; }
*createCommunicationLog(array $data, string $to, string $content, string $type = CommunicationLog::TYPE_EMAIL): CommunicationLog
**: Create a new communication log.
Example of define Communication log via model using ActivityLoggable
// Creating a communication log $communicationLog = $model->createCommunicationLog([ 'type' => 'cc' => ['cc@example.com'], 'bcc' => ['bcc@example.com'], 'subject' => 'Subject of the email', ], 'to@example.com', 'Content of the email');
Traits for activity log mailable to support tracking read email
Located in
src\Support\Traits\ReadMailableTrait.php
Using <activity-log-list>
or <v-activity-log>
to display activity log list. Pass filter as a slot if filter
functionality is needed
<ActivityLogList :model-id="tender.id" :model-class="tenderModel"> <v-filter entity="activity-logs" class="flex flex-row-reverse space-x-2 space-x-reverse"></v-filter> </ActivityLogList>
Events
The package provides the following events:
ActivityLogCommentCreated
: This event is fired when an activity log comment is created. The event receives the activity log instance.ActivityLogCommentDeleted
: This event is fired when an activity log comment is deleted. The event receives the activity log instance.ActivityLogCommentUpdated
: This event is fired when an activity log comment is update. The event receives the activity log instance.ActivityLogCommunicationRead
: This event is fired when an communication log is marked read. The event receives the activity log instance.
Usage
In order to log anything add the following trait to a model you want to log on.
... use Dcodegroup\ActivityLog\Support\Traits\ActivityLoggable; class Order extends Model { use ActivityLoggable; ... }
In addition, we can add activity log wherever we want the model
$model->createActivityLog([ 'type' => \Dcodegroup\ActivityLog\Models\ActivityLog::TYPE_COMMENT // if type is null default type will be TYPE_DATA, we support 3 other types: TYPE_STATUS, TYPE_COMMENT, TYPE_NOTIFICATION 'title' => 'left a comment.', 'description' => 'left a comment', ]);
Changelog
Please see CHANGELOG for more information about recent changes.
Contributing
We believe in the power of collaboration! If you share our passion for pushing the boundaries of business software, feel free to contribute, report issues, or suggest improvements. Your insights make us better.
Security
If you've found an issue related to this package that includes any security concerns; please email security@dcodegroup.com to ensure that we can prioritise the concerns in a confidential manner.
Credits
This project is supported & funded by Dcode Group and the team - both past and present. Special mention to:
About Dcode Group
Dcode Group specializes in crafting tailored software solutions utilizing the Laravel framework. Our focus lies in developing business, financial, and process-driven systems designed to support unique business operations. Leveraging packages like this one, we streamline common features/functions across projects, ensuring swift integration of broad functionalities while enhancing overall code base maintenance and management. Find out more about