This plugin implements an "audit trail" for any of your Table classes in your application, that is, the ability of recording any creation, modification or delete of the entities of any particular table.
By default, this plugin stores the audit logs into Elasticsearch, as we have found that it is a fantastic storage engine for append-only streams of data and provides really powerful features for finding changes in the historic data.
Even though we suggest storing the logs in Elasticsearch, this plugin is generic enough so you can implement your own persisting strategies, if so you wish.
You can install this plugin into your CakePHP application using composer and executing the following lines in the root of your application.
composer require lorenzo/audit-stash
bin/cake plugin load AuditStash
For using the default storage engine (ElasticSearch) you need to install the official elastic-search
plugin, by executing
the following lines:
composer require cakephp/elastic-search
bin/cake plugin load Cake/ElasticSearch
You now need to add the datasource configuration to your config/app.php
file:
'Datasources' => [
'auditlog_elastic' => [
'className' => 'Cake\ElasticSearch\Datasource\Connection',
'driver' => 'Cake\ElasticSearch\Datasource\Connection',
'host' => '127.0.0.1', // server where elasticsearch is running
'port' => 9200,
'index' => 'audit-logs', // The name of the index where to store the data
],
...
]
This is the most basic setup, it will use a single index for storing all the data. If you prefer to use a different index for each day (like logstash does), use the following configuration:
'Datasources' => [
'auditlog_elastic' => [
'className' => 'Cake\ElasticSearch\Datasource\Connection',
'driver' => 'Cake\ElasticSearch\Datasource\Connection',
'host' => '127.0.0.1', // server where elasticsearch is running
'port' => 9200,
'index' => 'audit-logs%s', // Just add a %s at the end
],
...
]
The added %s
will be used as a placeholder for the day of the year. Using an index per day is strongly recommended.
Enabling the Audit Log in any of your table classes is as simple as adding a behavior in the initialize()
function:
class ArticlesTable extends Table
{
public function initialize(array $config = [] )
{
...
$this->addBehavior('AuditStash.AuditLog');
}
}
When using the Elasticserch
persister, it is recommended that you tell Elasticsearch about the schema of your table. You can do this
automatically by executing the following command:
bin/cake elastic_mapping Articles
If you are using one index per day, save yourself some time and add the --use-templates
option. This will create a schema template so
any new index will inherit this configuration:
bin/cake elastic_mapping Articles --use-templates
Remember to execute the command line each time you change the schema of your table!
The AuditLog
behavior can be configured to ignore certain fields of your table, by default it ignores the created
and modified
fields:
class ArticlesTable extends Table
{
public function initialize(array $config = [] )
{
...
$this->addBehavior('AuditStash.AuditLog', [
'blacklist' => ['created', 'modified', 'another_field_name']
]);
}
}
If you prefer, you can use a whitelist
instead. This means that only the fields listed in that array will be tracked by the behavior:
public function initialize(array $config = [] )
{
...
$this->addBehavior('AuditStash.AuditLog', [
'whitelist' => ['title', 'description', 'author_id']
]);
}
It is often useful to store the identifier of the user that is triggering the changes in a certain table. For this purpose, AuditStash
provides the RequestMetadata
listener class, that is capable of storing the current URL, IP and logged in user. You need to add this
listener to your application in the AppController::beforeFilter()
method:
use AuditStash\Meta\RequestMetadata;
...
class AppController extends Controller
{
public function beforeFilter(Event $event)
{
...
$eventManager = $this->loadModel()->eventManager();
$eventManager->on(new RequestMetadata($this->request, $this->Auth->user('id')));
}
}
The above code assumes that you will trigger the table operations from the controller, using the default Table class for the controller. If you plan to use other Table classes for saving or deleting inside the same controller, it is advised that you attach the listener globally:
use AuditStash\Meta\RequestMetadata;
use Cake\Event\EventManager;
...
class AppController extends Controller
{
public function beforeFilter(Event $event)
{
...
EventManager::instance()->on(new RequestMetadata($this->request, $this->Auth->user('id')));
}
}
AuditStash
is also capable of storing arbitrary data for each of the logged events. You can use the ApplicationMetadata
listener or
create your own. If you choose to use ApplicationMetadata
, your logs will contain the app_name
key stored and any extra information
your may have provided. You can configure this listener anywhere in your application, such as the bootstrap.php
file or, again, directly
in your AppController.
use AuditStash\Meta\ApplicationMetadata;
use Cake\Event\EventManager;
EventManager::instance()->on(new ApplicationMetadata('my_blog_app', [
'server' => $theServerID,
'extra' => $somExtraInformation,
'moon_phase' => $currentMoonPhase
]));
Implementing your own metadata listeners is as simple as attaching the listener to the AuditStash.beforeLog
event. For example:
EventManager::instance()->on('AuditStash.beforeLog', function ($event, array $logs) {
foreach ($logs as $event) {
$event->setMetaInfo($event->getMetaInfo() + ['extra' => 'This is extra data to be stored']);
}
});
There are valid reasons for wanting to use a different persist engine for your audit logs. Luckily, this plugin allows you to implement
your own storage engines. It is as simple as implementing the PersisterInterface
interface:
use AuditStash\PersisterInterface;
class DatabasePersister implements PersisterInterface
{
public function logEvents(array $auditLogs)
{
foreach ($auditLogs as $log) {
$eventType = $log->getEventType();
$data = [
'timestamp' => $log->getTimestamp(),
'transaction' => $log->getTransactionId(),
'type' => $log->getEventType(),
'primary_key' => $log->getId(),
'source' => $log->getSourceName(),
'parent_source' => $log->getParentSourceName(),
'original' => json_encode($log->getOriginal()),
'changed' => $eventType === 'delete' ? null : json_encode($log->getChanged()),
'meta' => json_encode($log->getMetaInfo())
];
TableRegistry::get('MyAuditsTable')->save(new Entity($data));
}
}
}
The code above assumes you have a MyAuditsTable
class that is capable of persisting the structure if the
passed data.
Finally, you need to configure AuditStash
to use your new persister. In the config/app.php
file add the following
lines:
...
'AuditStash' => ['persister' => 'App\Namespace\For\Your\DatabasePersister']
or if you are using as standalone via
\Cake\Core\Configure::write('AuditStash.persister', 'App\Namespace\For\Your\DatabasePersister');
The configuration contains the fully namespaced class name of your persister.
Occasionally, you may want to wrap a number of database changes in a transaction, so that it can be rolled back if one part of the process fails. In order to create audit logs during a transaction, some additional setup is required. First create the file src/Model/Audit/AuditTrail.php
with the following:
<?php
namespace App\Model\Audit;
use Cake\Utility\Text;
use SplObjectStorage;
class AuditTrail
{
protected $_auditQueue;
protected $_auditTransaction;
public function __construct()
{
$this->_auditQueue = new SplObjectStorage;
$this->_auditTransaction = Text::uuid();
}
public function toSaveOptions()
{
return [
'_auditQueue' => $this->_auditQueue,
'_auditTransaction' => $this->_auditTransaction
];
}
}
Anywhere you wish to use Connection::transactional()
, you will need to first include the following at the top of the file:
use App\Model\Audit\AuditTrail;
use Cake\Event\Event;
Your transaction should then look similar to this example of a BookmarksController:
$trail = new AuditTrail();
$success = $this->Bookmarks->connection()->transactional(function () use ($trail) {
$bookmark = $this->Bookmarks->newEntity();
$bookmark1->save($data1, $trail->toSaveOptions());
$bookmark2 = $this->Bookmarks->newEntity();
$bookmark2->save($data2, $trail->toSaveOptions());
....
$bookmarkN = $this->Bookmarks->newEntity();
$bookmarkN->save($dataN, $trail->toSaveOptions());
return true;
});
if ($success) {
$event = new Event('Model.afterCommit', $this->Bookmarks);
$table->behaviors()->get('AuditLog')->afterCommit($event, $result, $auditTrail->toSaveOptions());
}
This will save all audit info for your objects, as well as audits for any associated data. Please note, $result
must be an instance of an Object. Do not change the text "Model.afterCommit".