lav45 / yii2-activity-logger
Tools to store user activity log for Yii2
Installs: 35 786
Dependents: 0
Suggesters: 0
Security: 0
Stars: 31
Watchers: 5
Forks: 15
Open Issues: 6
Type:yii2-extension
Requires
- php: >=5.5.0
- yiisoft/yii2: 2.0.*
Requires (Dev)
- ext-sqlite3: *
- phpunit/phpunit: 4.8 - 7.5
- roave/security-advisories: dev-master
README
Установка расширения
~$ composer require --prefer-dist lav45/yii2-activity-logger
Миграции
Для начала нужно настроить MigrateController
, таким образом чтобы он получал миграции из нескольких источников.
В настройках консольного окружения необходимо добавить следующий код:
return [ 'controllerMap' => [ 'migrate' => [ 'class' => \yii\console\controllers\MigrateController::class, 'migrationPath' => [ '@app/migrations', '@vendor/lav45/yii2-activity-logger/migrations', ], ], ], ];
Запускаем миграции
~$ yii migrate
Подключение
Необходимо добавить в конфигурационный файл
return [ 'modules' => [ /** * Модуль будет использоваться для просмотра логов */ 'logger' => [ 'class' => \lav45\activityLogger\modules\Module::class, // Список моделей которые логировались 'entityMap' => [ 'news' => 'common\models\News', ], ] ], 'components' => [ /** * Компонент принимает и управляет логами */ 'activityLogger' => [ 'class' => \lav45\activityLogger\Manager::class, // Включаем логирование для PROD версии 'enabled' => YII_ENV_PROD, // идентификатор компонента `\yii\web\User` 'user' => 'user', // Поле для отображения имени из модели пользователя 'userNameAttribute' => 'username', // Префикс нужен для того чтобы исключить пересечения userId если у вас в проекте авторизуется несколько разных сущностей 'userIdPrefix' => 'u', // идентификатор компонента хранилища логов `\lav45\activityLogger\StorageInterface` 'storage' => 'activityLoggerStorage', ], /** * Компонент принимает и управляет логами */ 'activityLoggerStorage' => [ 'class' => \lav45\activityLogger\DbStorage::class, // Имя таблицы в которой будут хранится логи 'tableName' => '{{%activity_log}}', // идентификатор компонента `\yii\db\Connection` 'db' => 'db', ], ] ];
Значения по умолчанию для всех лог записей можна задать через Yii::$container
Для удобства этот код можно разместить в файле bootstrap.php
Yii::$container->set(\lav45\activityLogger\LogMessageDTO::class, [ 'env' => 'console', // Окружение из которого проиводило действие 'userId' => 'console', 'userName' => 'Droid R2-D2', ]);
Ссылки для просмотра логов
// На этой странице можно просмотреть все логи Url::toRoute(['/logger/default/index']); // На этой странице можно просмотреть журналы действий конкретного пользователя по го `$id` Url::toRoute(['/logger/default/index', 'userId' => 1]); // На этой странице можно просмотреть журналы действий для всех объектов "news" Url::toRoute(['/logger/default/index', 'entityName' => 'news']); // На этой странице можно просмотреть журналы действий для всех объектов "news" с "id" => 1 Url::toRoute(['/logger/default/index', 'entityName' => 'news', 'entityId' => 1]);
Пример использования для ActiveRecord модели
/** * @mixin \lav45\activityLogger\ActiveLogBehavior */ class News extends ActiveRecord { // Рекомендуется использовать public function transactions() { return [ ActiveRecord::SCENARIO_DEFAULT => ActiveRecord::OP_ALL, ]; } public function behaviors() { return [ [ 'class' => \lav45\activityLogger\ActiveLogBehavior::class, // Если необхадимо изменить стандартное значение `entityName` 'getEntityName' => function () { return static::tableName(); }, // Если необхадимо изменить стандартное значение `entityId` 'getEntityId' => function () { return $this->getPrimaryKey(); } /** * В случаях когда нужно для конкретного ActiveLogBehavior делать подпись с понятным названием. * Если на странице выводятся история изменения всех пользователей, * то невсегда понятно у кого именно изменился статут, день рождения или другие данные */ 'beforeSaveMessage' => function ($data) { return ['attribute' => 'custom data'] + $data; } // Список полей за изменением которых будет производиться слежение 'attributes' => [ // Простые поля ( string|int|bool ) 'name', // Поля значение которого можно найти в списке. // в данном случае `$model->getStatusList()[$model->status]` 'status' => [ 'list' => 'statusList', ], // Поле значение которого является `id` связи с другой моделью 'template_id' => [ 'relation' => 'template', // Поле из связанной таблицы которое будет использовано в качестве отображаемого значения 'attribute' => 'name', ], ] ], [ /** * В случаях когда нужно для всех логов делать подпись с понятным названием. * Если на странице выводятся история изменения всех пользователей, * то невсегда понятно у кого именно изменился статут, день рождения или другие данные */ 'class' => \lav45\activityLogger\LogInfoBehavior::class, 'template' => '{username} ({profile.email})', ], ]; } /** * Если необхадимо форматировать отобоажемое значение * Можно указать любой поддерживаемый формат компонентом `\yii\i18n\Formatter` * или использовать произвольную функцию * @return array */ public function attributeFormats() { return [ // Значение аттрибуда будет форматироваться с помощью Yii::$app->formatter->asDatetime($value); 'published_at' => 'datetime', // Можно использовать свою функцию обратного вызова 'is_published' => function($value) { return Yii::$app->formatter->asBoolean($value); }, // Если нужно вывести имени картинки и ссылку на неё 'image' => function($value) { if (empty($value)) { return null; } $url = "https://cdn.site.com/img/{$value}"; return Html::a($value, $url, ['target' => '_blank']); } ]; } /** * В процессе работы `\lav45\activityLogger\ActiveLogBehavior` вызывает событие * [[ActiveLogBehavior::EVENT_BEFORE_SAVE_MESSAGE]] - перед записью логов * [[ActiveLogBehavior::EVENT_AFTER_SAVE_MESSAGE]] - после записи логов */ public function init() { parent::init(); // Регистрируем обработчики событий $this->on(ActiveLogBehavior::EVENT_BEFORE_SAVE_MESSAGE, function (\lav45\activityLogger\MessageEvent $event) { // Вы можете добавить в список логов свою информацию $event->logData[] = 'Reset password'; }); $this->on(ActiveLogBehavior::EVENT_AFTER_SAVE_MESSAGE, function (\yii\base\Event $event) { // Какие-то действия после записи логов }); } /* * В место регистрации события вы можете создать однаименный метод который будет вызываться вместо события */ /** * Будет вызываться в место события [[ActiveLogBehavior::EVENT_BEFORE_SAVE_MESSAGE]] * @return array */ public function beforeSaveMessage($data) { // Вы можете добавить в список логов свою информацию $data[] = 'Reset password'; // или заменить отображаемое значение в логах для атрибута `password_hash` $data['password_hash'] = 'Reset password'; return $data; } /** * Будет вызываться в место события [[ActiveLogBehavior::EVENT_AFTER_SAVE_MESSAGE]] */ public function afterSaveMessage() { // Какие-то действия после записи логов } }
Добавим консольный контроллер для очистки логов
Это не обязательное расширение. Если вы не планируете удалять устаревшие логи, можете пропустить этот пункт.
return [ 'controllerMap' => [ 'logger' => [ 'class' => \lav45\activityLogger\console\DefaultController::class ] ], ];
Если необходимо удалить старые логи для этого используйте консольный контроллер:
~$ yii logger/clean --old-than=1y # => Deleted 5 record(s) from the activity log.
Параметры командной строки
-
--entity-id, -eid
: string. Идентификатор целевого объекта -
--entity-name, -e
: string. Псевдоним имени целевого объекта -
--user-id, -uid
: string. Идентификатор пользователя, который выполнил действие -
--log-action, -a
: string. Действие, которое было произведено над объектом -
--env
: string. Среда, из которой производилось действие -
--old-than, -o
: string. Удаление старых данныхДопустимые значения:
- 1h - старше 1 часа
- 2d - старше 2 дней
- 3m - старше 3-х месяцев
- 4y - старше 4 лет.
Ручное использование компонента
Добавление логов
Пригодится в тех случаях когда в процессе работы приложения не используются ActiveRecord модели. Например при отправке отчетов, скачивании файлов, работа с внешним API, логирование процесса работы консольного контроллера и т.д
use lav45\activityLogger\LogMessageDTO; $message = Yii::createObject([ 'class' => LogMessageDTO::class, // имя сущности 'entityName' => 'user', // id сущности с которой производится действие 'entityId' => 10, // Действие которое сейчас выполняется 'action' => 'download', // текст с описанием действия 'data' => ['export data'], ]); Yii::$app->activityLogger->log($message);
Когда в логах нужна оставить одну запись со списком выполненных действий можно воспользоваться LogCollection
use lav45\activityLogger\LogCollection; $collection = new LogCollection(Yii::$app->activityLogger, 'entityName'); /** * Добавляем все необходимые записи */ $collection->addMessage('Created: 100'); $collection->addMessage('Updated: 100500'); $collection->addMessage('Deleted: 5'); /** * Сохраняем все собранные сообщение как одну запись в логах * Посли записи список логов будет очищен */ $collection->push(); // => true
Тестирование
~$ docker build --pull --build-arg UID=$(id -u) --build-arg GID=$(id -g) --rm -t php74-test .
~$ ./container composer update --prefer-dist
~$ ./container vendor/bin/phpunit
Лицензии
Для получения информации о лицензии проверьте файл LICENSE.md.