• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

codeigniter4 / CodeIgniter4 / 12739985319

13 Jan 2025 03:15AM UTC coverage: 84.486%. Remained the same
12739985319

push

github

paulbalandan
Merge branch 'develop' into 4.6

351 of 417 new or added lines in 121 files covered. (84.17%)

1 existing line in 1 file now uncovered.

20798 of 24617 relevant lines covered (84.49%)

189.94 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

67.63
/system/Session/Handlers/RedisHandler.php
1
<?php
2

3
declare(strict_types=1);
4

5
/**
6
 * This file is part of CodeIgniter 4 framework.
7
 *
8
 * (c) CodeIgniter Foundation <admin@codeigniter.com>
9
 *
10
 * For the full copyright and license information, please view
11
 * the LICENSE file that was distributed with this source code.
12
 */
13

14
namespace CodeIgniter\Session\Handlers;
15

16
use CodeIgniter\I18n\Time;
17
use CodeIgniter\Session\Exceptions\SessionException;
18
use Config\Session as SessionConfig;
19
use Redis;
20
use RedisException;
21
use ReturnTypeWillChange;
22

23
/**
24
 * Session handler using Redis for persistence
25
 */
26
class RedisHandler extends BaseHandler
27
{
28
    private const DEFAULT_PORT     = 6379;
29
    private const DEFAULT_PROTOCOL = 'tcp';
30

31
    /**
32
     * phpRedis instance
33
     *
34
     * @var Redis|null
35
     */
36
    protected $redis;
37

38
    /**
39
     * Key prefix
40
     *
41
     * @var string
42
     */
43
    protected $keyPrefix = 'ci_session:';
44

45
    /**
46
     * Lock key
47
     *
48
     * @var string|null
49
     */
50
    protected $lockKey;
51

52
    /**
53
     * Key exists flag
54
     *
55
     * @var bool
56
     */
57
    protected $keyExists = false;
58

59
    /**
60
     * Number of seconds until the session ends.
61
     *
62
     * @var int
63
     */
64
    protected $sessionExpiration = 7200;
65

66
    /**
67
     * Time (microseconds) to wait if lock cannot be acquired.
68
     */
69
    private int $lockRetryInterval = 100_000;
70

71
    /**
72
     * Maximum number of lock acquisition attempts.
73
     */
74
    private int $lockMaxRetries = 300;
75

76
    /**
77
     * @param string $ipAddress User's IP address
78
     *
79
     * @throws SessionException
80
     */
81
    public function __construct(SessionConfig $config, string $ipAddress)
82
    {
83
        parent::__construct($config, $ipAddress);
15✔
84

85
        // Store Session configurations
86
        $this->sessionExpiration = ($config->expiration === 0)
15✔
87
            ? (int) ini_get('session.gc_maxlifetime')
×
88
            : $config->expiration;
15✔
89

90
        // Add sessionCookieName for multiple session cookies.
91
        $this->keyPrefix .= $config->cookieName . ':';
15✔
92

93
        $this->setSavePath();
15✔
94

95
        if ($this->matchIP === true) {
15✔
96
            $this->keyPrefix .= $this->ipAddress . ':';
×
97
        }
98

99
        $this->lockRetryInterval = $config->lockWait ?? $this->lockRetryInterval;
15✔
100
        $this->lockMaxRetries    = $config->lockAttempts ?? $this->lockMaxRetries;
15✔
101
    }
102

103
    protected function setSavePath(): void
104
    {
105
        if (empty($this->savePath)) {
15✔
106
            throw SessionException::forEmptySavepath();
×
107
        }
108

109
        $url   = parse_url($this->savePath);
15✔
110
        $query = [];
15✔
111

112
        if ($url === false) {
15✔
113
            // Unix domain socket like `unix:///var/run/redis/redis.sock?persistent=1`.
114
            if (preg_match('#unix://(/[^:?]+)(\?.+)?#', $this->savePath, $matches)) {
1✔
115
                $host = $matches[1];
1✔
116
                $port = 0;
1✔
117

118
                if (isset($matches[2])) {
1✔
119
                    parse_str(ltrim($matches[2], '?'), $query);
×
120
                }
121
            } else {
122
                throw SessionException::forInvalidSavePathFormat($this->savePath);
×
123
            }
124
        } else {
125
            // Also accepts `/var/run/redis.sock` for backward compatibility.
126
            if (isset($url['path']) && $url['path'][0] === '/') {
14✔
127
                $host = $url['path'];
1✔
128
                $port = 0;
1✔
129
            } else {
130
                // TCP connection.
131
                if (! isset($url['host'])) {
13✔
132
                    throw SessionException::forInvalidSavePathFormat($this->savePath);
×
133
                }
134

135
                $protocol = $url['scheme'] ?? self::DEFAULT_PROTOCOL;
13✔
136
                $host     = $protocol . '://' . $url['host'];
13✔
137
                $port     = $url['port'] ?? self::DEFAULT_PORT;
13✔
138
            }
139

140
            if (isset($url['query'])) {
14✔
141
                parse_str($url['query'], $query);
5✔
142
            }
143
        }
144

145
        $password = $query['auth'] ?? null;
15✔
146
        $database = isset($query['database']) ? (int) $query['database'] : 0;
15✔
147
        $timeout  = isset($query['timeout']) ? (float) $query['timeout'] : 0.0;
15✔
148
        $prefix   = $query['prefix'] ?? null;
15✔
149

150
        $this->savePath = [
15✔
151
            'host'     => $host,
15✔
152
            'port'     => $port,
15✔
153
            'password' => $password,
15✔
154
            'database' => $database,
15✔
155
            'timeout'  => $timeout,
15✔
156
        ];
15✔
157

158
        if ($prefix !== null) {
15✔
159
            $this->keyPrefix = $prefix;
×
160
        }
161
    }
162

163
    /**
164
     * Re-initialize existing session, or creates a new one.
165
     *
166
     * @param string $path The path where to store/retrieve the session
167
     * @param string $name The session name
168
     *
169
     * @throws RedisException
170
     */
171
    public function open($path, $name): bool
172
    {
173
        if (empty($this->savePath)) {
6✔
174
            return false;
×
175
        }
176

177
        $redis = new Redis();
6✔
178

179
        if (
180
            ! $redis->connect(
6✔
181
                $this->savePath['host'],
6✔
182
                $this->savePath['port'],
6✔
183
                $this->savePath['timeout'],
6✔
184
            )
6✔
185
        ) {
186
            $this->logger->error('Session: Unable to connect to Redis with the configured settings.');
×
187
        } elseif (isset($this->savePath['password']) && ! $redis->auth($this->savePath['password'])) {
6✔
188
            $this->logger->error('Session: Unable to authenticate to Redis instance.');
×
189
        } elseif (isset($this->savePath['database']) && ! $redis->select($this->savePath['database'])) {
6✔
190
            $this->logger->error(
×
NEW
191
                'Session: Unable to select Redis database with index ' . $this->savePath['database'],
×
192
            );
×
193
        } else {
194
            $this->redis = $redis;
6✔
195

196
            return true;
6✔
197
        }
198

199
        return false;
×
200
    }
201

202
    /**
203
     * Reads the session data from the session storage, and returns the results.
204
     *
205
     * @param string $id The session ID
206
     *
207
     * @return false|string Returns an encoded string of the read data.
208
     *                      If nothing was read, it must return false.
209
     *
210
     * @throws RedisException
211
     */
212
    #[ReturnTypeWillChange]
213
    public function read($id)
214
    {
215
        if (isset($this->redis) && $this->lockSession($id)) {
4✔
216
            if (! isset($this->sessionID)) {
4✔
217
                $this->sessionID = $id;
4✔
218
            }
219

220
            $data = $this->redis->get($this->keyPrefix . $id);
4✔
221

222
            if (is_string($data)) {
4✔
223
                $this->keyExists = true;
2✔
224
            } else {
225
                $data = '';
2✔
226
            }
227

228
            $this->fingerprint = md5($data);
4✔
229

230
            return $data;
4✔
231
        }
232

233
        return false;
×
234
    }
235

236
    /**
237
     * Writes the session data to the session storage.
238
     *
239
     * @param string $id   The session ID
240
     * @param string $data The encoded session data
241
     *
242
     * @throws RedisException
243
     */
244
    public function write($id, $data): bool
245
    {
246
        if (! isset($this->redis)) {
1✔
247
            return false;
×
248
        }
249

250
        if ($this->sessionID !== $id) {
1✔
251
            if (! $this->releaseLock() || ! $this->lockSession($id)) {
×
252
                return false;
×
253
            }
254

255
            $this->keyExists = false;
×
256
            $this->sessionID = $id;
×
257
        }
258

259
        if (isset($this->lockKey)) {
1✔
260
            $this->redis->expire($this->lockKey, 300);
1✔
261

262
            if ($this->fingerprint !== ($fingerprint = md5($data)) || $this->keyExists === false) {
1✔
263
                if ($this->redis->set($this->keyPrefix . $id, $data, $this->sessionExpiration)) {
1✔
264
                    $this->fingerprint = $fingerprint;
1✔
265
                    $this->keyExists   = true;
1✔
266

267
                    return true;
1✔
268
                }
269

270
                return false;
×
271
            }
272

273
            return $this->redis->expire($this->keyPrefix . $id, $this->sessionExpiration);
×
274
        }
275

276
        return false;
×
277
    }
278

279
    /**
280
     * Closes the current session.
281
     */
282
    public function close(): bool
283
    {
284
        if (isset($this->redis)) {
4✔
285
            try {
286
                $pingReply = $this->redis->ping();
4✔
287

288
                if (($pingReply === true) || ($pingReply === '+PONG')) {
4✔
289
                    if (isset($this->lockKey) && ! $this->releaseLock()) {
4✔
290
                        return false;
×
291
                    }
292

293
                    if (! $this->redis->close()) {
4✔
294
                        return false;
4✔
295
                    }
296
                }
297
            } catch (RedisException $e) {
×
298
                $this->logger->error('Session: Got RedisException on close(): ' . $e->getMessage());
×
299
            }
300

301
            $this->redis = null;
4✔
302

303
            return true;
4✔
304
        }
305

306
        return true;
×
307
    }
308

309
    /**
310
     * Destroys a session
311
     *
312
     * @param string $id The session ID being destroyed
313
     *
314
     * @throws RedisException
315
     */
316
    public function destroy($id): bool
317
    {
318
        if (isset($this->redis, $this->lockKey)) {
×
319
            if (($result = $this->redis->del($this->keyPrefix . $id)) !== 1) {
×
320
                $this->logger->debug(
×
NEW
321
                    'Session: Redis::del() expected to return 1, got ' . var_export($result, true) . ' instead.',
×
322
                );
×
323
            }
324

325
            return $this->destroyCookie();
×
326
        }
327

328
        return false;
×
329
    }
330

331
    /**
332
     * Cleans up expired sessions.
333
     *
334
     * @param int $max_lifetime Sessions that have not updated
335
     *                          for the last max_lifetime seconds will be removed.
336
     *
337
     * @return false|int Returns the number of deleted sessions on success, or false on failure.
338
     */
339
    #[ReturnTypeWillChange]
340
    public function gc($max_lifetime)
341
    {
342
        return 1;
1✔
343
    }
344

345
    /**
346
     * Acquires an emulated lock.
347
     *
348
     * @param string $sessionID Session ID
349
     *
350
     * @throws RedisException
351
     */
352
    protected function lockSession(string $sessionID): bool
353
    {
354
        $lockKey = $this->keyPrefix . $sessionID . ':lock';
4✔
355

356
        // PHP 7 reuses the SessionHandler object on regeneration,
357
        // so we need to check here if the lock key is for the
358
        // correct session ID.
359
        if ($this->lockKey === $lockKey) {
4✔
360
            // If there is the lock, make the ttl longer.
361
            return $this->redis->expire($this->lockKey, 300);
×
362
        }
363

364
        $attempt = 0;
4✔
365

366
        do {
367
            $result = $this->redis->set(
4✔
368
                $lockKey,
4✔
369
                (string) Time::now()->getTimestamp(),
4✔
370
                // NX -- Only set the key if it does not already exist.
371
                // EX seconds -- Set the specified expire time, in seconds.
372
                ['nx', 'ex' => 300],
4✔
373
            );
4✔
374

375
            if (! $result) {
4✔
376
                usleep($this->lockRetryInterval);
×
377

378
                continue;
×
379
            }
380

381
            $this->lockKey = $lockKey;
4✔
382
            break;
4✔
383
        } while (++$attempt < $this->lockMaxRetries);
×
384

385
        if ($attempt === 300) {
4✔
386
            $this->logger->error(
×
387
                'Session: Unable to obtain lock for ' . $this->keyPrefix . $sessionID
×
NEW
388
                . ' after 300 attempts, aborting.',
×
389
            );
×
390

391
            return false;
×
392
        }
393

394
        $this->lock = true;
4✔
395

396
        return true;
4✔
397
    }
398

399
    /**
400
     * Releases a previously acquired lock
401
     *
402
     * @throws RedisException
403
     */
404
    protected function releaseLock(): bool
405
    {
406
        if (isset($this->redis, $this->lockKey) && $this->lock) {
4✔
407
            if (! $this->redis->del($this->lockKey)) {
4✔
408
                $this->logger->error('Session: Error while trying to free lock for ' . $this->lockKey);
×
409

410
                return false;
×
411
            }
412

413
            $this->lockKey = null;
4✔
414
            $this->lock    = false;
4✔
415
        }
416

417
        return true;
4✔
418
    }
419
}
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc