. // }}} namespace Component\Notification; use App\Core\DB\DB; use App\Core\Event; use function App\Core\I18n\_m; use App\Core\Log; use App\Core\Modules\Component; use App\Core\Queue\Queue; use App\Core\Router\RouteLoader; use App\Core\Router\Router; use App\Entity\Activity; use App\Entity\Actor; use App\Entity\LocalUser; use App\Util\Exception\ServerException; use Component\FreeNetwork\FreeNetwork; use Component\Notification\Controller\Feed; use Exception; use Throwable; class Notification extends Component { public function onAddRoute(RouteLoader $m): bool { $m->connect('feed_notifications', '/feed/notifications', [Feed::class, 'notifications']); return Event::next; } /** * @throws ServerException */ public function onCreateDefaultFeeds(int $actor_id, LocalUser $user, int &$ordering): bool { DB::persist(\App\Entity\Feed::create([ 'actor_id' => $actor_id, 'url' => Router::url($route = 'feed_notifications'), 'route' => $route, 'title' => _m('Notifications'), 'ordering' => $ordering++, ])); return Event::next; } /** * Enqueues a notification for an Actor (such as person or group) which means * it shows up in their home feed and such. */ public function onNewNotification(Actor $sender, Activity $activity, array $ids_already_known = [], ?string $reason = null): bool { $targets = $activity->getNotificationTargets(ids_already_known: $ids_already_known, sender_id: $sender->getId()); if (Event::handle('NewNotificationWithTargets', [$sender, $activity, $targets, $reason]) === Event::next) { self::notify($sender, $activity, $targets, $reason); } return Event::next; } public function onQueueNotificationLocal(Actor $sender, Activity $activity, Actor $target, ?string $reason, array &$retry_args): bool { // TODO: use https://symfony.com/doc/current/notifier.html return Event::stop; } public function onQueueNotificationRemote(Actor $sender, Activity $activity, array $targets, ?string $reason, array &$retry_args): bool { if (FreeNetwork::notify($sender, $activity, $targets, $reason)) { return Event::stop; } else { return Event::next; } } /** * Bring given Activity to Targets's attention * * @return true if successful, false otherwise */ public static function notify(Actor $sender, Activity $activity, array $targets, ?string $reason = null): bool { $remote_targets = []; foreach ($targets as $target) { if ($target->getIsLocal()) { if ($target->hasBlocked($activity->getActor())) { Log::info("Not saving reply to actor {$target->getId()} from sender {$sender->getId()} because of a block."); continue; } if (Event::handle('NewNotificationShould', [$activity, $target]) === Event::next) { if ($sender->getId() === $target->getId() || $activity->getActorId() === $target->getId()) { // The target already knows about this, no need to bother with a notification continue; } } Queue::enqueue( payload: [$sender, $activity, $target, $reason], queue: 'notification_local', priority: true, ); } else { // We have no authority nor responsibility of notifying remote actors of a remote actor's doing if ($sender->getIsLocal()) { $remote_targets[] = $target; } } // XXX: Unideal as in failures the rollback will leave behind a false notification, // but most notifications (all) require flushing the objects first // Should be okay as long as implementors bear this in mind try { DB::wrapInTransaction(fn () => DB::persist(Entity\Notification::create([ 'activity_id' => $activity->getId(), 'target_id' => $target->getId(), 'reason' => $reason, ]))); } catch (Exception|Throwable $e) { // We do our best not to record duplicated notifications, but it's not insane that can happen Log::error('It was attempted to record an invalid notification!', [$e]); } } if ($remote_targets !== []) { Queue::enqueue( payload: [$sender, $activity, $remote_targets, $reason], queue: 'notification_remote', priority: false, ); } return true; } }