Completed
Pull Request — master (#125)
by Alexander
05:18
created

BasicNetworkResolver::process()   B

Complexity

Conditions 11
Paths 14

Size

Total Lines 35
Code Lines 24

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 23
CRAP Score 11.0619

Importance

Changes 2
Bugs 0 Features 2
Metric Value
cc 11
eloc 24
c 2
b 0
f 2
nc 14
nop 2
dl 0
loc 35
ccs 23
cts 25
cp 0.92
crap 11.0619
rs 7.3166

How to fix   Complexity   

Long Method

Small methods make your code easier to understand, in particular if combined with a good name. Besides, if your method is small, finding a good name is usually much easier.

For example, if you find yourself adding comments to a method's body, this is usually a good sign to extract the commented part to a new method, and use the comment as a starting point when coming up with a good name for this new method.

Commonly applied refactorings include:

1
<?php
2
3
4
namespace Yiisoft\Yii\Web\Middleware;
5
6
7
use Psr\Http\Message\ResponseInterface;
8
use Psr\Http\Message\ServerRequestInterface;
9
use Psr\Http\Server\MiddlewareInterface;
10
use Psr\Http\Server\RequestHandlerInterface;
11
12
/**
13
 * Basic network resolver updates an instance of server request with protocol from special headers.
14
 *
15
 * It can be used in the following cases:
16
 * - not required IP resolve to access the user's IP
17
 * - user's IP is already resolved (eg `ngx_http_realip_module` or similar)
18
 */
19
class BasicNetworkResolver implements MiddlewareInterface
20
{
21
    private const DEFAULT_PROTOCOL_AND_ACCEPTABLE_VALUES = [
22
        'http' => ['http'],
23
        'https' => ['https', 'on'],
24
    ];
25
26
    private $protocolHeaders = [];
27
28 14
    public function process(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface
29
    {
30 14
        $newScheme = null;
31 14
        foreach ($this->protocolHeaders as $header => $data) {
32 10
            if (!$request->hasHeader($header)) {
33 1
                continue;
34
            }
35 9
            $headerValues = $request->getHeader($header);
36 9
            if (is_callable($data)) {
37 2
                $newScheme = $data($headerValues, $header, $request);
38 2
                if ($newScheme === null) {
39 1
                    continue;
40
                }
41 1
                if (!is_string($newScheme)) {
42
                    throw new \RuntimeException('The scheme is neither string nor null!');
43
                }
44 1
                if ($newScheme === '') {
45
                    throw new \RuntimeException('The scheme cannot be an empty string!');
46
                }
47 1
                break;
48
            }
49 7
            $headerValue = strtolower($headerValues[0]);
50 7
            foreach ($data as $protocol => $acceptedValues) {
51 7
                if (!in_array($headerValue, $acceptedValues, true)) {
52 2
                    continue;
53
                }
54 6
                $newScheme = $protocol;
55 6
                break 2;
56
            }
57
        }
58 14
        $uri = $request->getUri();
59 14
        if ($newScheme !== null && $newScheme !== $uri->getScheme()) {
60 7
            $request = $request->withUri($uri->withScheme($newScheme));
61
        }
62 14
        return $handler->handle($request);
63
    }
64
65
    /**
66
     * With added header to check for determining whether the connection is made via HTTP or HTTPS (or any protocol).
67
     *
68
     * The match of header names and values is case-insensitive.
69
     * It's not advisable to put insecure/untrusted headers here.
70
     *
71
     * Accepted types of values:
72
     * - NULL (default): {{DEFAULT_PROTOCOL_AND_ACCEPTABLE_VALUES}}
73
     * - callable: custom function for getting the protocol
74
     * ```php
75
     * ->withProtocolHeader('x-forwarded-proto', function(array $values, string $header, ServerRequestInterface $request) {
76
     *   return $values[0] === 'https' ? 'https' : 'http';
77
     *   return null;     // If it doesn't make sense.
78
     * });
79
     * ```
80
     * - array: The array keys are protocol string and the array value is a list of header values that indicate the protocol.
81
     * ```php
82
     * ->withProtocolHeader('x-forwarded-proto', [
83
     *   'http' => ['http'],
84
     *   'https' => ['https']
85
     * ]);
86
     * ```
87
     * @param string $header
88
     * @param callable|array|null $values
89
     * @see DEFAULT_PROTOCOL_AND_ACCEPTABLE_VALUES
90
     */
91 12
    public function withAddedProtocolHeader(string $header, $values = null): self
92
    {
93 12
        $new = clone $this;
94 12
        $header = strtolower($header);
95 12
        if ($values === null) {
96 4
            $new->protocolHeaders[$header] = self::DEFAULT_PROTOCOL_AND_ACCEPTABLE_VALUES;
97 4
            return $new;
98
        }
99 8
        if (is_callable($values)) {
100 2
            $new->protocolHeaders[$header] = $values;
101 2
            return $new;
102
        }
103 6
        if (!is_array($values)) {
104
            throw new \RuntimeException('Accepted values is not array nor callable!');
105
        }
106 6
        if (count($values) === 0) {
107
            throw new \RuntimeException('Accepted values cannot be an empty array!');
108
        }
109 6
        $new->protocolHeaders[$header] = [];
110 6
        foreach ($values as $protocol => $acceptedValues) {
111 6
            if (!is_string($protocol)) {
112
                throw new \RuntimeException('The protocol must be type of string!');
113
            }
114 6
            if ($protocol === '') {
115
                throw new \RuntimeException('The protocol cannot be an empty string!');
116
            }
117 6
            $new->protocolHeaders[$header][$protocol] = array_map('strtolower', (array)$acceptedValues);
118
        }
119 6
        return $new;
120
    }
121
122 2
    public function withoutProtocolHeader(string $header): self
123
    {
124 2
        $new = clone $this;
125 2
        unset($new->protocolHeaders[strtolower($header)]);
126 2
        return $new;
127
    }
128
129 2
    public function withoutProtocolHeaders(?array $headers = null): self
130
    {
131 2
        $new = clone $this;
132 2
        if ($headers === null) {
133 1
            $new->protocolHeaders = [];
134
        } else {
135 1
            foreach ($headers as $header) {
136 1
                $new = $new->withoutProtocolHeader($header);
137
            }
138
        }
139 2
        return $new;
140
    }
141
}
142