Completed
Pull Request — master (#25)
by Dawid
04:04 queued 02:14
created

Router::__construct()   A

Complexity

Conditions 1
Paths 1

Size

Total Lines 3
Code Lines 1

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 2
CRAP Score 1

Importance

Changes 0
Metric Value
cc 1
eloc 1
nc 1
nop 0
dl 0
loc 3
ccs 2
cts 2
cp 1
crap 1
rs 10
c 0
b 0
f 0
1
<?php declare(strict_types=1);
2
3
namespace Igni\Http\Router;
4
5
use Igni\Http\Exception\GenericHttpException;;
6
use Igni\Http\Route;
0 ignored issues
show
Bug introduced by
This use statement conflicts with another class in this namespace, Igni\Http\Router\Route. Consider defining an alias.

Let?s assume that you have a directory layout like this:

.
|-- OtherDir
|   |-- Bar.php
|   `-- Foo.php
`-- SomeDir
    `-- Foo.php

and let?s assume the following content of Bar.php:

// Bar.php
namespace OtherDir;

use SomeDir\Foo; // This now conflicts the class OtherDir\Foo

If both files OtherDir/Foo.php and SomeDir/Foo.php are loaded in the same runtime, you will see a PHP error such as the following:

PHP Fatal error:  Cannot use SomeDir\Foo as Foo because the name is already in use in OtherDir/Foo.php

However, as OtherDir/Foo.php does not necessarily have to be loaded and the error is only triggered if it is loaded before OtherDir/Bar.php, this problem might go unnoticed for a while. In order to prevent this error from surfacing, you must import the namespace with a different alias:

// Bar.php
namespace OtherDir;

use SomeDir\Foo as SomeDirFoo; // There is no conflict anymore.
Loading history...
7
use Igni\Http\Router as RouterInterface;
8
use Symfony\Component\Routing\Route as SymfonyRoute;
9
use Symfony\Component\Routing\Exception\MethodNotAllowedException as SymfonyMethodNotAllowedException;
10
use Symfony\Component\Routing\Exception\ResourceNotFoundException;
11
use Symfony\Component\Routing\Matcher\UrlMatcher;
12
use Symfony\Component\Routing\RequestContext;
13
use Symfony\Component\Routing\RouteCollection;
14
15
/**
16
 * Wrapper for package symfony/router
17
 *
18
 * @package Igni\Http\Router
19
 */
20
class Router implements RouterInterface
21
{
22
    /** @var RouteCollection */
23
    protected $routeCollection;
24
25
    /** @var Route[] */
26
    protected $routes = [];
27
28 20
    public function __construct()
29
    {
30 20
        $this->routeCollection = new RouteCollection();
31 20
    }
32
33
    /**
34
     * Registers new route.
35
     *
36
     * @param Route $route
37
     */
38 15
    public function addRoute(Route $route): void
39
    {
40 15
        if ($route instanceof \Igni\Http\Router\Route) {
41 15
            $name = $route->getName();
42
        } else {
43
            $name = \Igni\Http\Router\Route::generateNameFromPath($route->getPath());
44
        }
45
46 15
        $baseRoute = new SymfonyRoute($route->getPath());
47 15
        $baseRoute->setMethods($route->getMethods());
48
49 15
        $this->routeCollection->add($name, $baseRoute);
50 15
        $this->routes[$name] = $route;
51 15
    }
52
53
    /**
54
     * Finds route matching clients request.
55
     *
56
     * @param string $method request method.
57
     * @param string $path request path.
58
     * @return Route
59
     */
60 16
    public function findRoute(string $method, string $path): Route
61
    {
62 16
        $matcher = new UrlMatcher($this->routeCollection, new RequestContext('/', $method));
63
        try {
64 16
            $route = $matcher->match($path);
65 4
        } catch (ResourceNotFoundException $exception) {
66 3
            throw GenericHttpException::invalidUri($path, $method);
67 1
        } catch (SymfonyMethodNotAllowedException $exception) {
68 1
            throw GenericHttpException::methodNotAllowed($path, $method, $exception->getAllowedMethods());
69
        }
70
71 12
        $routeName = $route['_route'];
72 12
        unset($route['_route']);
73
74 12
        return $this->routes[$routeName]->withAttributes($route);
75
    }
76
}
77