2021-12-12 01:29:33 +08:00
|
|
|
<?php
|
|
|
|
|
2023-05-18 00:56:55 +08:00
|
|
|
namespace BookStack\Activity\Tools;
|
2021-12-12 01:29:33 +08:00
|
|
|
|
2023-05-24 16:06:15 +08:00
|
|
|
use BookStack\Activity\DispatchWebhookJob;
|
2023-05-18 00:56:55 +08:00
|
|
|
use BookStack\Activity\Models\Activity;
|
|
|
|
use BookStack\Activity\Models\Loggable;
|
|
|
|
use BookStack\Activity\Models\Webhook;
|
2023-07-19 18:03:05 +08:00
|
|
|
use BookStack\Activity\Notifications\NotificationManager;
|
2021-12-12 01:29:33 +08:00
|
|
|
use BookStack\Entities\Models\Entity;
|
2022-07-17 20:28:56 +08:00
|
|
|
use BookStack\Facades\Theme;
|
|
|
|
use BookStack\Theming\ThemeEvents;
|
2021-12-12 06:29:33 +08:00
|
|
|
use Illuminate\Database\Eloquent\Builder;
|
2021-12-12 01:29:33 +08:00
|
|
|
use Illuminate\Support\Facades\Log;
|
|
|
|
|
|
|
|
class ActivityLogger
|
|
|
|
{
|
2023-07-19 18:03:05 +08:00
|
|
|
public function __construct(
|
|
|
|
protected NotificationManager $notifications
|
|
|
|
) {
|
|
|
|
$this->notifications->loadDefaultHandlers();
|
|
|
|
}
|
|
|
|
|
2021-12-12 01:29:33 +08:00
|
|
|
/**
|
|
|
|
* Add a generic activity event to the database.
|
|
|
|
*/
|
2023-07-19 18:03:05 +08:00
|
|
|
public function add(string $type, string|Loggable $detail = ''): void
|
2021-12-12 01:29:33 +08:00
|
|
|
{
|
|
|
|
$detailToStore = ($detail instanceof Loggable) ? $detail->logDescriptor() : $detail;
|
|
|
|
|
|
|
|
$activity = $this->newActivityForUser($type);
|
|
|
|
$activity->detail = $detailToStore;
|
|
|
|
|
|
|
|
if ($detail instanceof Entity) {
|
2024-05-04 23:28:18 +08:00
|
|
|
$activity->loggable_id = $detail->id;
|
|
|
|
$activity->loggable_type = $detail->getMorphClass();
|
2021-12-12 01:29:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
$activity->save();
|
2022-07-17 20:28:56 +08:00
|
|
|
|
2021-12-12 01:29:33 +08:00
|
|
|
$this->setNotification($type);
|
2021-12-12 06:29:33 +08:00
|
|
|
$this->dispatchWebhooks($type, $detail);
|
2023-08-15 21:39:39 +08:00
|
|
|
$this->notifications->handle($activity, $detail, user());
|
2022-07-17 20:28:56 +08:00
|
|
|
Theme::dispatch(ThemeEvents::ACTIVITY_LOGGED, $type, $detail);
|
2021-12-12 01:29:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a new activity instance for the current user.
|
|
|
|
*/
|
|
|
|
protected function newActivityForUser(string $type): Activity
|
|
|
|
{
|
|
|
|
return (new Activity())->forceFill([
|
|
|
|
'type' => strtolower($type),
|
|
|
|
'user_id' => user()->id,
|
2022-07-23 20:41:29 +08:00
|
|
|
'ip' => IpFormatter::fromCurrentRequest()->format(),
|
2021-12-12 01:29:33 +08:00
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes the entity attachment from each of its activities
|
|
|
|
* and instead uses the 'extra' field with the entities name.
|
|
|
|
* Used when an entity is deleted.
|
|
|
|
*/
|
2023-07-18 22:07:31 +08:00
|
|
|
public function removeEntity(Entity $entity): void
|
2021-12-12 01:29:33 +08:00
|
|
|
{
|
|
|
|
$entity->activity()->update([
|
2024-05-04 23:28:18 +08:00
|
|
|
'detail' => $entity->name,
|
|
|
|
'loggable_id' => null,
|
|
|
|
'loggable_type' => null,
|
2021-12-12 01:29:33 +08:00
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Flashes a notification message to the session if an appropriate message is available.
|
|
|
|
*/
|
2021-12-12 06:29:33 +08:00
|
|
|
protected function setNotification(string $type): void
|
2021-12-12 01:29:33 +08:00
|
|
|
{
|
|
|
|
$notificationTextKey = 'activities.' . $type . '_notification';
|
|
|
|
if (trans()->has($notificationTextKey)) {
|
|
|
|
$message = trans($notificationTextKey);
|
|
|
|
session()->flash('success', $message);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-18 22:07:31 +08:00
|
|
|
protected function dispatchWebhooks(string $type, string|Loggable $detail): void
|
2021-12-12 06:29:33 +08:00
|
|
|
{
|
2021-12-13 01:39:06 +08:00
|
|
|
$webhooks = Webhook::query()
|
2021-12-18 19:43:05 +08:00
|
|
|
->whereHas('trackedEvents', function (Builder $query) use ($type) {
|
2021-12-13 01:39:06 +08:00
|
|
|
$query->where('event', '=', $type)
|
|
|
|
->orWhere('event', '=', 'all');
|
|
|
|
})
|
|
|
|
->where('active', '=', true)
|
|
|
|
->get();
|
2021-12-12 06:29:33 +08:00
|
|
|
|
|
|
|
foreach ($webhooks as $webhook) {
|
|
|
|
dispatch(new DispatchWebhookJob($webhook, $type, $detail));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-12 01:29:33 +08:00
|
|
|
/**
|
|
|
|
* Log out a failed login attempt, Providing the given username
|
|
|
|
* as part of the message if the '%u' string is used.
|
|
|
|
*/
|
2023-07-18 22:07:31 +08:00
|
|
|
public function logFailedLogin(string $username): void
|
2021-12-12 01:29:33 +08:00
|
|
|
{
|
|
|
|
$message = config('logging.failed_login.message');
|
|
|
|
if (!$message) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$message = str_replace('%u', $username, $message);
|
|
|
|
$channel = config('logging.failed_login.channel');
|
|
|
|
Log::channel($channel)->warning($message);
|
|
|
|
}
|
|
|
|
}
|