anyRequestIncludesTheAllowAccessHeader()   A
last analyzed

Complexity

Conditions 1
Paths 1

Size

Total Lines 18
Code Lines 12

Duplication

Lines 0
Ratio 0 %

Importance

Changes 1
Bugs 0 Features 0
Metric Value
eloc 12
c 1
b 0
f 0
dl 0
loc 18
rs 9.8666
cc 1
nc 1
nop 0
1
<?php
2
3
declare(strict_types=1);
4
5
namespace ShlinkioTest\Shlink\Rest\Middleware;
6
7
use Laminas\Diactoros\Response;
8
use Laminas\Diactoros\ServerRequest;
9
use Mezzio\Router\Route;
10
use Mezzio\Router\RouteResult;
11
use PHPUnit\Framework\TestCase;
12
use Prophecy\Argument;
13
use Prophecy\PhpUnit\ProphecyTrait;
14
use Prophecy\Prophecy\ObjectProphecy;
15
use Psr\Http\Server\RequestHandlerInterface;
16
use Shlinkio\Shlink\Rest\Middleware\CrossDomainMiddleware;
17
18
use function Laminas\Stratigility\middleware;
19
20
class CrossDomainMiddlewareTest extends TestCase
21
{
22
    use ProphecyTrait;
23
24
    private CrossDomainMiddleware $middleware;
25
    private ObjectProphecy $handler;
26
27
    public function setUp(): void
28
    {
29
        $this->middleware = new CrossDomainMiddleware();
30
        $this->handler = $this->prophesize(RequestHandlerInterface::class);
31
    }
32
33
    /** @test */
34
    public function nonCrossDomainRequestsAreNotAffected(): void
35
    {
36
        $originalResponse = (new Response())->withStatus(404);
37
        $this->handler->handle(Argument::any())->willReturn($originalResponse)->shouldBeCalledOnce();
38
39
        $response = $this->middleware->process(new ServerRequest(), $this->handler->reveal());
40
        $headers = $response->getHeaders();
41
42
        self::assertSame($originalResponse, $response);
43
        self::assertEquals(404, $response->getStatusCode());
44
        self::assertArrayNotHasKey('Access-Control-Allow-Origin', $headers);
45
        self::assertArrayNotHasKey('Access-Control-Expose-Headers', $headers);
46
        self::assertArrayNotHasKey('Access-Control-Allow-Methods', $headers);
47
        self::assertArrayNotHasKey('Access-Control-Max-Age', $headers);
48
        self::assertArrayNotHasKey('Access-Control-Allow-Headers', $headers);
49
    }
50
51
    /** @test */
52
    public function anyRequestIncludesTheAllowAccessHeader(): void
53
    {
54
        $originalResponse = new Response();
55
        $this->handler->handle(Argument::any())->willReturn($originalResponse)->shouldBeCalledOnce();
56
57
        $response = $this->middleware->process(
58
            (new ServerRequest())->withHeader('Origin', 'local'),
59
            $this->handler->reveal(),
60
        );
61
        self::assertNotSame($originalResponse, $response);
62
63
        $headers = $response->getHeaders();
64
65
        self::assertEquals('local', $response->getHeaderLine('Access-Control-Allow-Origin'));
66
        self::assertEquals('X-Api-Key', $response->getHeaderLine('Access-Control-Expose-Headers'));
67
        self::assertArrayNotHasKey('Access-Control-Allow-Methods', $headers);
68
        self::assertArrayNotHasKey('Access-Control-Max-Age', $headers);
69
        self::assertArrayNotHasKey('Access-Control-Allow-Headers', $headers);
70
    }
71
72
    /** @test */
73
    public function optionsRequestIncludesMoreHeaders(): void
74
    {
75
        $originalResponse = new Response();
76
        $request = (new ServerRequest())
77
            ->withMethod('OPTIONS')
78
            ->withHeader('Origin', 'local')
79
            ->withHeader('Access-Control-Request-Headers', 'foo, bar, baz');
80
        $this->handler->handle(Argument::any())->willReturn($originalResponse)->shouldBeCalledOnce();
81
82
        $response = $this->middleware->process($request, $this->handler->reveal());
83
        self::assertNotSame($originalResponse, $response);
84
85
        $headers = $response->getHeaders();
86
87
        self::assertEquals('local', $response->getHeaderLine('Access-Control-Allow-Origin'));
88
        self::assertEquals('X-Api-Key', $response->getHeaderLine('Access-Control-Expose-Headers'));
89
        self::assertArrayHasKey('Access-Control-Allow-Methods', $headers);
90
        self::assertEquals('1000', $response->getHeaderLine('Access-Control-Max-Age'));
91
        self::assertEquals('foo, bar, baz', $response->getHeaderLine('Access-Control-Allow-Headers'));
92
        self::assertEquals(204, $response->getStatusCode());
93
    }
94
95
    /**
96
     * @test
97
     * @dataProvider provideRouteResults
98
     */
99
    public function optionsRequestParsesRouteMatchToDetermineAllowedMethods(
100
        ?RouteResult $result,
101
        string $expectedAllowedMethods
102
    ): void {
103
        $originalResponse = new Response();
104
        $request = (new ServerRequest())->withAttribute(RouteResult::class, $result)
105
                                        ->withMethod('OPTIONS')
106
                                        ->withHeader('Origin', 'local');
107
        $this->handler->handle(Argument::any())->willReturn($originalResponse)->shouldBeCalledOnce();
108
109
        $response = $this->middleware->process($request, $this->handler->reveal());
110
111
        self::assertEquals($response->getHeaderLine('Access-Control-Allow-Methods'), $expectedAllowedMethods);
112
        self::assertEquals(204, $response->getStatusCode());
113
    }
114
115
    public function provideRouteResults(): iterable
116
    {
117
        yield 'with no route result' => [null, 'GET,POST,PUT,PATCH,DELETE,OPTIONS'];
118
        yield 'with failed route result' => [RouteResult::fromRouteFailure(['POST', 'GET']), 'POST,GET'];
119
        yield 'with success route result' => [
120
            RouteResult::fromRoute(
121
                new Route('/', middleware(function (): void {
122
                }), ['DELETE', 'PATCH', 'PUT']),
123
            ),
124
            'DELETE,PATCH,PUT',
125
        ];
126
    }
127
128
    /**
129
     * @test
130
     * @dataProvider provideMethods
131
     */
132
    public function expectedStatusCodeIsReturnDependingOnRequestMethod(
133
        string $method,
134
        int $status,
135
        int $expectedStatus
136
    ): void {
137
        $originalResponse = (new Response())->withStatus($status);
138
        $request = (new ServerRequest())->withMethod($method)
139
                                        ->withHeader('Origin', 'local');
140
        $this->handler->handle(Argument::any())->willReturn($originalResponse)->shouldBeCalledOnce();
141
142
        $response = $this->middleware->process($request, $this->handler->reveal());
143
144
        self::assertEquals($expectedStatus, $response->getStatusCode());
145
    }
146
147
    public function provideMethods(): iterable
148
    {
149
        yield 'POST 200' => ['POST', 200, 200];
150
        yield 'POST 400' => ['POST', 400, 400];
151
        yield 'POST 500' => ['POST', 500, 500];
152
        yield 'GET 200' => ['GET', 200, 200];
153
        yield 'GET 400' => ['GET', 400, 400];
154
        yield 'GET 500' => ['GET', 500, 500];
155
        yield 'PATCH 200' => ['PATCH', 200, 200];
156
        yield 'PATCH 400' => ['PATCH', 400, 400];
157
        yield 'PATCH 500' => ['PATCH', 500, 500];
158
        yield 'DELETE 200' => ['DELETE', 200, 200];
159
        yield 'DELETE 400' => ['DELETE', 400, 400];
160
        yield 'DELETE 500' => ['DELETE', 500, 500];
161
        yield 'OPTIONS 200' => ['OPTIONS', 200, 204];
162
        yield 'OPTIONS 400' => ['OPTIONS', 400, 204];
163
        yield 'OPTIONS 500' => ['OPTIONS', 500, 204];
164
    }
165
}
166