Completed
Pull Request — master (#5)
by Sebastian
01:48
created

CsrfGuard::tokenIsValid()   A

Complexity

Conditions 3
Paths 3

Size

Total Lines 14
Code Lines 6

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 6
CRAP Score 3

Importance

Changes 0
Metric Value
dl 0
loc 14
ccs 6
cts 6
cp 1
rs 9.4285
c 0
b 0
f 0
cc 3
eloc 6
nc 3
nop 3
crap 3
1
<?php
2
3
/**
4
 * Linna Cross-site Request Forgery Guard
5
 *
6
 * @author Sebastian Rapetti <[email protected]>
7
 * @copyright (c) 2017, Sebastian Rapetti
8
 * @license http://opensource.org/licenses/MIT MIT License
9
 */
10
declare(strict_types=1);
11
12
namespace Linna;
13
14
use RuntimeException;
15
16
/**
17
 * Cross-site Request Forgery Guard
18
 */
19
class CsrfGuard
20
{
21
    /**
22
     * @var array Php session data from superglobal.
23
     */
24
    private $session;
25
26
    /**
27
     * @var int Max number of tokens stored in session.
28
     */
29
    private $maxStorage;
30
31
    /**
32
     * @var int Rapresent the lenght of the token in bytes.
33
     */
34
    private $tokenStrength;
35
    
36
    /**
37
     * __construct.
38
     *
39
     * Class constructor.
40
     *
41
     * @param int $maxStorage    Max number of tokens stored in session, work as
42
     *                           FIFO data structure, when maximun capacity is
43
     *                           reached, oldest token be dequeued from storage.
44
     * @param int $tokenStrength Rapresent the lenght of the token in bytes.
45
     */
46 23
    public function __construct(int $maxStorage, int $tokenStrength)
47
    {
48 23
        if (session_status() === 1) {
49 1
            throw new RuntimeException(__CLASS__.': Session must be started before create '.__CLASS__.' instance.');
50
        }
51
        
52
        //if csrf array doesn't exist inside session, initialize it.
53
        //for code shortness: Null coalescing operator
54
        //http://php.net/manual/en/migration70.new-features.php
55 22
        $_SESSION['CSRF'] = $_SESSION['CSRF'] ?? [];
56
57 22
        $this->session = &$_SESSION;
58 22
        $this->maxStorage = $maxStorage;
59 22
        $this->tokenStrength = $tokenStrength;
60 22
    }
61
62
    /**
63
     * dequeue.
64
     *
65
     * Limit number of token stored in session.
66
     */
67 21
    private function dequeue(array &$array)
68
    {
69 21
        $size = count($array);
70
        
71 21
        while ($size > $this->maxStorage) {
72 14
            array_shift($array);
73 14
            $size--;
74
        }
75 21
    }
76
77
    /**
78
     * getToken.
79
     *
80
     * Return csrf token as array.
81
     *
82
     * @return array
83
     */
84 18
    public function getToken() : array
85
    {
86 18
        $token = $this->generateToken();
87
88 18
        $name = $token['name'];
89
90 18
        $this->session['CSRF'][$name] = $token;
91
92
        //storage cleaning!
93
        //warning!! if you get in a page more token of maximun storage,
94
        //will there a leak of token, the firsts generated
95
        //in future I think throw and exception.
96 18
        $this->dequeue($this->session['CSRF']);
97
98 18
        return $token;
99
    }
100
101
    /**
102
     * getTimedToken.
103
     *
104
     * Return timed csrf token as array.
105
     *
106
     * @param int $ttl Time to live for the token.
107
     *
108
     * @return array
109
     */
110 3
    public function getTimedToken(int $ttl) : array
111
    {
112 3
        $token = $this->generateToken();
113 3
        $token['time'] = time() + $ttl;
114
        
115 3
        $name = $token['name'];
116
117 3
        $this->session['CSRF'][$name] = $token;
118
119 3
        $this->dequeue($this->session['CSRF']);
120
121 3
        return $token;
122
    }
123
    
124
    /**
125
     * generateToken.
126
     *
127
     * Generate a random token.
128
     *
129
     * @return array
130
     */
131 21
    private function generateToken() : array
132
    {
133 21
        $name = 'csrf_'.bin2hex(random_bytes(8));
134 21
        $value = bin2hex(random_bytes($this->tokenStrength));
135
        
136 21
        return ['name' => $name, 'value' => $value];
137
    }
138
    
139
    /**
140
     * getHiddenInput.
141
     *
142
     * Return csrf token as hidden input form.
143
     *
144
     * @return string
145
     *
146
     * @deprecated since version 1.1.0
147
     */
148 1
    public function getHiddenInput() : string
149
    {
150 1
        $token = $this->getToken();
151
152 1
        return '<input type="hidden" name="'.$token['name'].'" value="'.$token['value'].'" />';
153
    }
154
155
    /**
156
     * validate.
157
     *
158
     * Validate a csrf token or a csrf timed token.
159
     *
160
     * @param array $requestData From request or from superglobal variables $_POST,
161
     *                           $_GET, $_REQUEST and $_COOKIE.
162
     *
163
     * @return bool
164
     */
165 4
    public function validate(array $requestData) : bool
166
    {
167
        //apply matchToken method elements of passed data,
168
        //using this instead of forach for code shortness.
169 4
        $array = array_filter($requestData, array($this, 'doChecks'), ARRAY_FILTER_USE_BOTH);
170
171 4
        return (bool) count($array);
172
    }
173
    
174
    /**
175
     * doChecks.
176
     *
177
     * Tests for valid token.
178
     *
179
     * @param string $value
180
     * @param string $key
181
     *
182
     * @return bool
183
     */
184 4
    private function doChecks(string $value, string $key) : bool
185
    {
186 4
        $tokens = &$this->session['CSRF'];
187
188 4
        return $this->tokenIsValid($tokens, $value, $key) &&
189 4
               $this->tokenIsExiperd($tokens, $key)  &&
190 4
               $this->deleteToken($tokens, $key);
191
    }
192
193
    /**
194
     * deleteToken.
195
     *
196
     * Delete token after validation.
197
     *
198
     * @param array $tokens
199
     * @param string $key
200
     * @return bool
201
     */
202 3
    private function deleteToken(array &$tokens, string &$key) : bool
203
    {
204 3
        unset($tokens[$key]);
205
206 3
        return true;
207
    }
208
    
209
    /**
210
     * tokenIsValid.
211
     *
212
     * Check if token is valid
213
     *
214
     * @param array $tokens
215
     * @param string $value
216
     * @param string $key
217
     *
218
     * @return bool
219
     */
220 4
    private function tokenIsValid(array &$tokens, string &$value, string &$key) : bool
221
    {
222
        //if token exist
223 4
        if (!isset($tokens[$key])) {
224 2
            return false;
225
        }
226
227
        //if token has valid value
228 4
        if (!hash_equals($tokens[$key]['value'], $value)) {
229 1
            return false;
230
        }
231
232 4
        return true;
233
    }
234
235
    /**
236
     * tokenIsExiperd.
237
     *
238
     * Check if timed token is expired.
239
     *
240
     * @param array $tokens
241
     * @param string $key
242
     *
243
     * @return bool
244
     */
245 4
    private function tokenIsExiperd(array &$tokens, string &$key) : bool
246
    {
247
        //if timed and if time is valid
248 4
        if (isset($tokens[$key]['time']) && $tokens[$key]['time'] < time()) {
249 1
            return false;
250
        }
251
252 3
        return true;
253
    }
254
}
255