mirror of https://github.com/flightphp/core
				
				
				
			
			You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							347 lines
						
					
					
						
							12 KiB
						
					
					
				
			
		
		
	
	
							347 lines
						
					
					
						
							12 KiB
						
					
					
				<?php
 | 
						|
 | 
						|
declare(strict_types=1);
 | 
						|
 | 
						|
namespace tests;
 | 
						|
 | 
						|
use ArgumentCountError;
 | 
						|
use Closure;
 | 
						|
use Exception;
 | 
						|
use flight\core\Dispatcher;
 | 
						|
use flight\Engine;
 | 
						|
use InvalidArgumentException;
 | 
						|
use PharIo\Manifest\InvalidEmailException;
 | 
						|
use tests\classes\Hello;
 | 
						|
use PHPUnit\Framework\TestCase;
 | 
						|
use tests\classes\ClassWithExceptionInConstruct;
 | 
						|
use tests\classes\ContainerDefault;
 | 
						|
use tests\classes\TesterClass;
 | 
						|
use TypeError;
 | 
						|
 | 
						|
class DispatcherTest extends TestCase
 | 
						|
{
 | 
						|
    private Dispatcher $dispatcher;
 | 
						|
 | 
						|
    protected function setUp(): void
 | 
						|
    {
 | 
						|
        $this->dispatcher = new Dispatcher();
 | 
						|
    }
 | 
						|
 | 
						|
    public function testClosureMapping(): void
 | 
						|
    {
 | 
						|
        $this->dispatcher->set('map1', Closure::fromCallable(function (): string {
 | 
						|
            return 'hello';
 | 
						|
        }));
 | 
						|
 | 
						|
        $this->assertSame('hello', $this->dispatcher->run('map1'));
 | 
						|
    }
 | 
						|
 | 
						|
    public function testFunctionMapping(): void
 | 
						|
    {
 | 
						|
        $this->dispatcher->set('map2', fn (): string => 'hello');
 | 
						|
 | 
						|
        $this->assertSame('hello', $this->dispatcher->run('map2'));
 | 
						|
    }
 | 
						|
 | 
						|
    public function testHasEvent(): void
 | 
						|
    {
 | 
						|
        $this->dispatcher->set('map-event', function (): void {
 | 
						|
        });
 | 
						|
 | 
						|
        $this->assertTrue($this->dispatcher->has('map-event'));
 | 
						|
    }
 | 
						|
 | 
						|
    public function testClearAllRegisteredEvents(): void
 | 
						|
    {
 | 
						|
        $customFunction = $anotherFunction = function (): void {
 | 
						|
        };
 | 
						|
 | 
						|
        $this->dispatcher
 | 
						|
            ->set('map-event', $customFunction)
 | 
						|
            ->set('map-event-2', $anotherFunction);
 | 
						|
 | 
						|
        $this->assertTrue($this->dispatcher->has('map-event'));
 | 
						|
        $this->assertTrue($this->dispatcher->has('map-event-2'));
 | 
						|
 | 
						|
        $this->dispatcher->clear();
 | 
						|
 | 
						|
        $this->assertFalse($this->dispatcher->has('map-event'));
 | 
						|
        $this->assertFalse($this->dispatcher->has('map-event-2'));
 | 
						|
    }
 | 
						|
 | 
						|
    public function testClearDeclaredRegisteredEvent(): void
 | 
						|
    {
 | 
						|
        $customFunction = $anotherFunction = function (): void {
 | 
						|
        };
 | 
						|
 | 
						|
        $this->dispatcher
 | 
						|
            ->set('map-event', $customFunction)
 | 
						|
            ->set('map-event-2', $anotherFunction);
 | 
						|
 | 
						|
        $this->assertTrue($this->dispatcher->has('map-event'));
 | 
						|
        $this->assertTrue($this->dispatcher->has('map-event-2'));
 | 
						|
 | 
						|
        $this->dispatcher->clear('map-event');
 | 
						|
 | 
						|
        $this->assertFalse($this->dispatcher->has('map-event'));
 | 
						|
        $this->assertTrue($this->dispatcher->has('map-event-2'));
 | 
						|
    }
 | 
						|
 | 
						|
    public function testStaticFunctionMapping(): void
 | 
						|
    {
 | 
						|
        $this->dispatcher->set('map2', Hello::class . '::sayBye');
 | 
						|
 | 
						|
        $this->assertSame('goodbye', $this->dispatcher->run('map2'));
 | 
						|
    }
 | 
						|
 | 
						|
    public function testClassMethodMapping(): void
 | 
						|
    {
 | 
						|
        $this->dispatcher->set('map3', [new Hello(), 'sayHi']);
 | 
						|
 | 
						|
        $this->assertSame('hello', $this->dispatcher->run('map3'));
 | 
						|
    }
 | 
						|
 | 
						|
    public function testStaticClassMethodMapping(): void
 | 
						|
    {
 | 
						|
        $this->dispatcher->set('map4', [Hello::class, 'sayBye']);
 | 
						|
 | 
						|
        $this->assertSame('goodbye', $this->dispatcher->run('map4'));
 | 
						|
    }
 | 
						|
 | 
						|
    public function testBeforeAndAfter(): void
 | 
						|
    {
 | 
						|
        $this->dispatcher->set('hello', fn (string $name): string => "Hello, $name!");
 | 
						|
 | 
						|
        $this->dispatcher
 | 
						|
            ->hook('hello', Dispatcher::FILTER_BEFORE, function (array &$params): void {
 | 
						|
                // Manipulate the parameter
 | 
						|
                $params[0] = 'Fred';
 | 
						|
            })
 | 
						|
            ->hook('hello', Dispatcher::FILTER_AFTER, function (array &$params, string &$output): void {
 | 
						|
                // Manipulate the output
 | 
						|
                $output .= ' Have a nice day!';
 | 
						|
            });
 | 
						|
 | 
						|
        $result = $this->dispatcher->run('hello', ['Bob']);
 | 
						|
 | 
						|
        $this->assertSame('Hello, Fred! Have a nice day!', $result);
 | 
						|
    }
 | 
						|
 | 
						|
    public function testBeforeAndAfterWithShortAfterFilterSyntax(): void
 | 
						|
    {
 | 
						|
        $this->dispatcher->set('hello', fn (string $name): string => "Hello, $name!");
 | 
						|
 | 
						|
        $this->dispatcher
 | 
						|
            ->hook('hello', Dispatcher::FILTER_BEFORE, function (array &$params): void {
 | 
						|
                // Manipulate the parameter
 | 
						|
                $params[0] = 'Fred';
 | 
						|
            })
 | 
						|
            ->hook('hello', Dispatcher::FILTER_AFTER, function (string &$output): void {
 | 
						|
                // Manipulate the output
 | 
						|
                $output .= ' Have a nice day!';
 | 
						|
            });
 | 
						|
 | 
						|
        $result = $this->dispatcher->run('hello', ['Bob']);
 | 
						|
 | 
						|
        $this->assertSame('Hello, Fred! Have a nice day!', $result);
 | 
						|
    }
 | 
						|
 | 
						|
    public function testInvalidCallback(): void
 | 
						|
    {
 | 
						|
        $this->expectException(Exception::class);
 | 
						|
        $this->expectExceptionMessage("Class 'NonExistentClass' not found. Is it being correctly autoloaded with Flight::path()?");
 | 
						|
 | 
						|
        $this->dispatcher->execute(['NonExistentClass', 'nonExistentMethod']);
 | 
						|
    }
 | 
						|
 | 
						|
    public function testInvalidCallableString(): void
 | 
						|
    {
 | 
						|
        $this->expectException(InvalidArgumentException::class);
 | 
						|
        $this->expectExceptionMessage('Invalid callback specified.');
 | 
						|
 | 
						|
        $this->dispatcher->execute('nonexistentGlobalFunction');
 | 
						|
    }
 | 
						|
 | 
						|
    // It will be useful for executing instance Controller methods statically
 | 
						|
    public function testCanExecuteAnNonStaticMethodStatically(): void
 | 
						|
    {
 | 
						|
        $this->assertSame('hello', $this->dispatcher->execute([Hello::class, 'sayHi']));
 | 
						|
    }
 | 
						|
 | 
						|
    public function testItThrowsAnExceptionWhenRunAnUnregistedEventName(): void
 | 
						|
    {
 | 
						|
        $this->expectException(Exception::class);
 | 
						|
 | 
						|
        $this->dispatcher->run('nonExistentEvent');
 | 
						|
    }
 | 
						|
 | 
						|
    public function testWhenAnEventThrowsAnExceptionItPersistUntilNextCatchBlock(): void
 | 
						|
    {
 | 
						|
        $this->dispatcher->set('myMethod', function (): void {
 | 
						|
            throw new Exception('myMethod Exception');
 | 
						|
        });
 | 
						|
 | 
						|
        $this->expectException(Exception::class);
 | 
						|
        $this->expectExceptionMessage('myMethod Exception');
 | 
						|
 | 
						|
        $this->dispatcher->run('myMethod');
 | 
						|
    }
 | 
						|
 | 
						|
    public function testWhenAnEventThrowsCustomExceptionItPersistUntilNextCatchBlock(): void
 | 
						|
    {
 | 
						|
        $this->dispatcher->set('checkEmail', function (string $email): void {
 | 
						|
            throw new InvalidEmailException("Invalid email $email");
 | 
						|
        });
 | 
						|
 | 
						|
        $this->expectException(InvalidEmailException::class);
 | 
						|
        $this->expectExceptionMessage('Invalid email mail@mail,com');
 | 
						|
 | 
						|
        $this->dispatcher->run('checkEmail', ['mail@mail,com']);
 | 
						|
    }
 | 
						|
 | 
						|
    public function testItThrowsNoticeForOverrideRegisteredEvents(): void
 | 
						|
    {
 | 
						|
        set_error_handler(function (int $errno, string $errstr): void {
 | 
						|
            $this->assertSame(E_USER_NOTICE, $errno);
 | 
						|
            $this->assertSame("Event 'myMethod' has been overridden!", $errstr);
 | 
						|
        });
 | 
						|
 | 
						|
        $this->dispatcher->set('myMethod', function (): string {
 | 
						|
            return 'Original';
 | 
						|
        });
 | 
						|
 | 
						|
        $this->dispatcher->set('myMethod', function (): string {
 | 
						|
            return 'Overridden';
 | 
						|
        });
 | 
						|
 | 
						|
        $this->assertSame('Overridden', $this->dispatcher->run('myMethod'));
 | 
						|
        restore_error_handler();
 | 
						|
    }
 | 
						|
 | 
						|
    public function testItThrowsNoticeForInvalidFilterTypes(): void
 | 
						|
    {
 | 
						|
        set_error_handler(function (int $errno, string $errstr): void {
 | 
						|
            $this->assertSame(E_USER_NOTICE, $errno);
 | 
						|
            $this->assertStringStartsWith("Invalid filter type 'invalid', use ", $errstr);
 | 
						|
        });
 | 
						|
 | 
						|
        $this->dispatcher
 | 
						|
            ->set('myMethod', function (): string {
 | 
						|
                return 'Original';
 | 
						|
            })
 | 
						|
            ->hook('myMethod', 'invalid', function (array &$params, &$output): void {
 | 
						|
                $output = 'Overridden';
 | 
						|
            });
 | 
						|
 | 
						|
        $this->assertSame('Original', $this->dispatcher->run('myMethod'));
 | 
						|
        restore_error_handler();
 | 
						|
    }
 | 
						|
 | 
						|
    public function testItThrowsAnExceptionForInvalidFilters(): void
 | 
						|
    {
 | 
						|
        $this->expectException(Exception::class);
 | 
						|
        $this->expectExceptionMessage('Invalid callable $filters[1]');
 | 
						|
 | 
						|
        $params = [];
 | 
						|
        $output = '';
 | 
						|
        $invalidCallable = 'invalidGlobalFunction';
 | 
						|
 | 
						|
        $validCallable = function (): void {
 | 
						|
        };
 | 
						|
 | 
						|
        $this->dispatcher->filter([$validCallable, $invalidCallable], $params, $output);
 | 
						|
    }
 | 
						|
 | 
						|
    public function testCallFunction6Params(): void
 | 
						|
    {
 | 
						|
        $func = function ($param1, $param2, $param3, $param4, $param5, $param6) {
 | 
						|
            return "hello{$param1}{$param2}{$param3}{$param4}{$param5}{$param6}";
 | 
						|
        };
 | 
						|
 | 
						|
        $params = ['param1', 'param2', 'param3', 'param4', 'param5', 'param6'];
 | 
						|
        $result = $this->dispatcher->callFunction($func, $params);
 | 
						|
 | 
						|
        $this->assertSame('helloparam1param2param3param4param5param6', $result);
 | 
						|
    }
 | 
						|
 | 
						|
    public function testInvokeMethod(): void
 | 
						|
    {
 | 
						|
        $class = new TesterClass('param1', 'param2', 'param3', 'param4', 'param5', 'param6');
 | 
						|
        $result = $this->dispatcher->invokeMethod([$class, 'instanceMethod']);
 | 
						|
 | 
						|
        $this->assertSame('param1', $class->param2);
 | 
						|
    }
 | 
						|
 | 
						|
    public function testExecuteStringClassBadConstructParams(): void
 | 
						|
    {
 | 
						|
        $this->expectException(ArgumentCountError::class);
 | 
						|
        $this->expectExceptionMessageMatches('#Too few arguments to function tests\\\\classes\\\\TesterClass::__construct\(\), 1 passed .+ and exactly 6 expected#');
 | 
						|
        $this->dispatcher->execute(TesterClass::class . '->instanceMethod');
 | 
						|
    }
 | 
						|
 | 
						|
    public function testExecuteStringClassNoConstruct(): void
 | 
						|
    {
 | 
						|
        $result = $this->dispatcher->execute(Hello::class . '->sayHi');
 | 
						|
        $this->assertSame('hello', $result);
 | 
						|
    }
 | 
						|
 | 
						|
    public function testExecuteStringClassNoConstructDoubleColon(): void
 | 
						|
    {
 | 
						|
        $result = $this->dispatcher->execute(Hello::class . '::sayHi');
 | 
						|
        $this->assertSame('hello', $result);
 | 
						|
    }
 | 
						|
 | 
						|
    public function testExecuteStringClassNoConstructArraySyntax(): void
 | 
						|
    {
 | 
						|
        $result = $this->dispatcher->execute([Hello::class, 'sayHi']);
 | 
						|
        $this->assertSame('hello', $result);
 | 
						|
    }
 | 
						|
 | 
						|
    public function testExecuteStringClassDefaultContainer(): void
 | 
						|
    {
 | 
						|
        $engine = new Engine();
 | 
						|
        $engine->set('test_me_out', 'You got it boss!');
 | 
						|
        $this->dispatcher->setEngine($engine);
 | 
						|
        $result = $this->dispatcher->execute(ContainerDefault::class . '->testTheContainer');
 | 
						|
        $this->assertSame('You got it boss!', $result);
 | 
						|
    }
 | 
						|
 | 
						|
    public function testExecuteStringClassDefaultContainerDoubleColon(): void
 | 
						|
    {
 | 
						|
        $engine = new Engine();
 | 
						|
        $engine->set('test_me_out', 'You got it boss!');
 | 
						|
        $this->dispatcher->setEngine($engine);
 | 
						|
        $result = $this->dispatcher->execute(ContainerDefault::class . '::testTheContainer');
 | 
						|
        $this->assertSame('You got it boss!', $result);
 | 
						|
    }
 | 
						|
 | 
						|
    public function testExecuteStringClassDefaultContainerArraySyntax(): void
 | 
						|
    {
 | 
						|
        $engine = new Engine();
 | 
						|
        $engine->set('test_me_out', 'You got it boss!');
 | 
						|
        $this->dispatcher->setEngine($engine);
 | 
						|
        $result = $this->dispatcher->execute([ContainerDefault::class, 'testTheContainer']);
 | 
						|
        $this->assertSame('You got it boss!', $result);
 | 
						|
    }
 | 
						|
 | 
						|
    public function testExecuteStringClassDefaultContainerButForgotInjectingEngine(): void
 | 
						|
    {
 | 
						|
        $this->expectException(TypeError::class);
 | 
						|
        $this->expectExceptionMessageMatches('#tests\\\\classes\\\\ContainerDefault::__construct\(\).+flight\\\\Engine, null given#');
 | 
						|
        $result = $this->dispatcher->execute([ContainerDefault::class, 'testTheContainer']);
 | 
						|
    }
 | 
						|
 | 
						|
    public function testContainerDicePdoWrapperTestBadParams()
 | 
						|
    {
 | 
						|
        $dice = new \Dice\Dice();
 | 
						|
        $this->dispatcher->setContainerHandler(function ($class, $params) use ($dice) {
 | 
						|
            return $dice->create($class, $params);
 | 
						|
        });
 | 
						|
 | 
						|
        $this->expectException(Exception::class);
 | 
						|
        $this->expectExceptionMessage('This is an exception in the constructor');
 | 
						|
 | 
						|
        $this->dispatcher->invokeCallable([ ClassWithExceptionInConstruct::class, '__construct' ]);
 | 
						|
    }
 | 
						|
}
 |