1
0
mirror of https://github.com/flarum/core.git synced 2025-10-26 13:16:11 +01:00
Files
php-flarum/tests/integration/extenders/EventTest.php
Alexander Skvortsov 9b2d7856d1 Add subscribe method to event extender (#2535)
Historically, extensions using subscribers has caused problems because subscribers were constructed/applied at extension boot. This caused some classes (e.g. UrlGenerator) to be resolved early, breaking parts of Flarum. For this reason, subscriber support wasn't included in the initial version of the Event extender.

However, updating extensions has shown that there is a legitimate use case for subscribers in organizing clean code; for instance, core's own `DiscussionMetadataUpdater`.

This commit introduces support for subscribers, but only applies them after the app has booted, which avoids the early resolution issues. Since event listeners/subscribers are only intended to be used with domain events, which would never be dispatched during app boot, the late activation of subscribers should not cause issue.
2021-01-15 20:33:29 -05:00

142 lines
3.8 KiB
PHP

<?php
/*
* This file is part of Flarum.
*
* For detailed copyright and license information, please view the
* LICENSE file that was distributed with this source code.
*/
namespace Flarum\Tests\integration\extenders;
use Flarum\Extend;
use Flarum\Foundation\Application;
use Flarum\Group\Command\CreateGroup;
use Flarum\Group\Event\Created;
use Flarum\Tests\integration\RetrievesAuthorizedUsers;
use Flarum\Tests\integration\TestCase;
use Flarum\User\User;
use Illuminate\Contracts\Bus\Dispatcher as BusDispatcher;
use Illuminate\Contracts\Events\Dispatcher;
use Symfony\Component\Translation\TranslatorInterface;
class EventTest extends TestCase
{
use RetrievesAuthorizedUsers;
protected function buildGroup()
{
$bus = $this->app()->getContainer()->make(BusDispatcher::class);
return $bus->dispatch(
new CreateGroup(User::find(1), ['attributes' => [
'nameSingular' => 'test group',
'namePlural' => 'test groups',
'color' => '#000000',
'icon' => 'fas fa-crown',
]])
);
}
/**
* @test
*/
public function custom_listener_doesnt_work_by_default()
{
$group = $this->buildGroup();
$this->assertEquals('test group', $group->name_singular);
}
/**
* @test
*/
public function custom_listener_works_with_closure()
{
$this->extend((new Extend\Event)->listen(Created::class, function (Created $event) {
$event->group->name_singular = 'modified group';
}));
$group = $this->buildGroup();
$this->assertEquals('modified group', $group->name_singular);
}
/**
* @test
*/
public function custom_listener_works_with_class_with_handle_method_and_can_inject_stuff()
{
// Because it injects a translator, this also tests that stuff can be injected into this callback.
$this->extend((new Extend\Event)->listen(Created::class, CustomListener::class));
$group = $this->buildGroup();
$this->assertEquals('core.group.admin', $group->name_singular);
}
/**
* @test
*/
public function custom_subscriber_works()
{
// Because it injects a translator, this also tests that stuff can be injected into this callback.
$this->extend((new Extend\Event)->subscribe(CustomSubscriber::class));
$group = $this->buildGroup();
$this->assertEquals('core.group.admin', $group->name_singular);
}
/**
* @test
*/
public function custom_subscriber_applied_after_app_booted()
{
// Because it injects a translator, this also tests that stuff can be injected into this callback.
$this->extend((new Extend\Event)->subscribe(CustomSubscriber::class));
$group = $this->buildGroup();
$this->assertEquals('booted', $group->name_plural);
}
}
class CustomListener
{
protected $translator;
public function __construct(TranslatorInterface $translator)
{
$this->translator = $translator;
}
public function handle(Created $event)
{
$event->group->name_singular = $this->translator->trans('core.group.admin');
}
}
class CustomSubscriber
{
protected $bootedAtConstruct;
protected $translator;
public function __construct(Application $app, TranslatorInterface $translator)
{
$this->bootedAtConstruct = $app->isBooted();
$this->translator = $translator;
}
public function subscribe(Dispatcher $events)
{
$events->listen(Created::class, [$this, 'whenGroupCreated']);
}
public function whenGroupCreated(Created $event)
{
$event->group->name_singular = $this->translator->trans('core.group.admin');
$event->group->name_plural = $this->bootedAtConstruct ? 'booted' : 'not booted';
}
}