Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions .github/workflows/reusable-phpunit-test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,10 @@ on:
description: Additional PHP extensions that are needed to be enabled
type: string
required: false
extra-ini-options:
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This change to this file should go the develop branch in a separate PR as I believe it can be used early.

Copy link
Author

@sk757a sk757a Jan 9, 2026

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do you think we should make similar changes in the reusable-serviceless-phpunit-test.yml file?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, please

description: Additional PHP configuration directives that should be appended to the php.ini
type: string
required: false
extra-composer-options:
description: Additional Composer options that should be appended to the `composer update` call
type: string
Expand Down Expand Up @@ -163,6 +167,7 @@ jobs:
php-version: ${{ inputs.php-version }}
tools: composer
extensions: gd, ${{ inputs.extra-extensions }}
ini-values: ${{ inputs.extra-ini-options }}
coverage: ${{ env.COVERAGE_DRIVER }}
env:
COVERAGE_DRIVER: ${{ inputs.enable-coverage && 'xdebug' || 'none' }}
Expand Down
3 changes: 2 additions & 1 deletion .github/workflows/test-phpunit.yml
Original file line number Diff line number Diff line change
Expand Up @@ -163,7 +163,8 @@ jobs:
enable-artifact-upload: ${{ matrix.php-version == needs.coverage-php-version.outputs.version }}
enable-coverage: ${{ matrix.php-version == needs.coverage-php-version.outputs.version }}
enable-profiling: ${{ matrix.php-version == needs.coverage-php-version.outputs.version }}
extra-extensions: redis, memcached
extra-extensions: redis, memcached, apcu
extra-ini-options: apc.enable_cli=1
extra-composer-options: ${{ matrix.composer-option }}

coveralls:
Expand Down
1 change: 1 addition & 0 deletions admin/framework/composer.json
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@
"predis/predis": "^3.0"
},
"suggest": {
"ext-apcu": "If you use Cache class ApcuHandler",
"ext-curl": "If you use CURLRequest class",
"ext-dom": "If you use TestResponse",
"ext-exif": "If you run Image class tests",
Expand Down
2 changes: 2 additions & 0 deletions app/Config/Cache.php
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
namespace Config;

use CodeIgniter\Cache\CacheInterface;
use CodeIgniter\Cache\Handlers\ApcuHandler;
use CodeIgniter\Cache\Handlers\DummyHandler;
use CodeIgniter\Cache\Handlers\FileHandler;
use CodeIgniter\Cache\Handlers\MemcachedHandler;
Expand Down Expand Up @@ -143,6 +144,7 @@ class Cache extends BaseConfig
* @var array<string, class-string<CacheInterface>>
*/
public array $validHandlers = [
'apcu' => ApcuHandler::class,
'dummy' => DummyHandler::class,
'file' => FileHandler::class,
'memcached' => MemcachedHandler::class,
Expand Down
1 change: 1 addition & 0 deletions composer.json
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@
"codeigniter4/framework": "self.version"
},
"suggest": {
"ext-apcu": "If you use Cache class ApcuHandler",
"ext-curl": "If you use CURLRequest class",
"ext-dom": "If you use TestResponse",
"ext-exif": "If you run Image class tests",
Expand Down
163 changes: 163 additions & 0 deletions system/Cache/Handlers/ApcuHandler.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,163 @@
<?php

declare(strict_types=1);

/**
* This file is part of CodeIgniter 4 framework.
*
* (c) CodeIgniter Foundation <[email protected]>
*
* For the full copyright and license information, please view
* the LICENSE file that was distributed with this source code.
*/

namespace CodeIgniter\Cache\Handlers;

use APCUIterator;
use Closure;
use CodeIgniter\I18n\Time;
use Config\Cache;

/**
* APCu cache handler
*
* @see \CodeIgniter\Cache\Handlers\ApcuHandlerTest
*/
class ApcuHandler extends BaseHandler
{
/**
* Note: Use `CacheFactory::getHandler()` to instantiate.
*/
public function __construct(Cache $config)
{
$this->prefix = $config->prefix;
}

public function initialize(): void
{
}

/**
* {@inheritDoc}
*/
public function get(string $key): mixed
{
$key = static::validateKey($key, $this->prefix);
$success = false;

$data = apcu_fetch($key, $success);

// Success returned by reference from apcu_fetch()
return $success ? $data : null;
}

/**
* {@inheritDoc}
*/
public function save(string $key, $value, int $ttl = 60): bool
{
$key = static::validateKey($key, $this->prefix);

return apcu_store($key, $value, $ttl);
}

/**
* {@inheritDoc}
*/
public function remember(string $key, int $ttl, Closure $callback): mixed
{
$key = static::validateKey($key, $this->prefix);

return apcu_entry($key, $callback, $ttl);
}

/**
* {@inheritDoc}
*/
public function delete(string $key): bool
{
$key = static::validateKey($key, $this->prefix);

return apcu_delete($key);
}

/**
* {@inheritDoc}
*/
public function deleteMatching(string $pattern): int
{
$matchedKeys = array_filter(
array_keys(iterator_to_array(new APCUIterator(null, APC_ITER_KEY))),
static fn ($key): bool => fnmatch($pattern, $key),
);

if ($matchedKeys !== []) {
return count($matchedKeys) - count(apcu_delete($matchedKeys));
}

return 0;
}

/**
* {@inheritDoc}
*/
public function increment(string $key, int $offset = 1): false|int
{
$key = static::validateKey($key, $this->prefix);

return apcu_inc($key, $offset);
}

/**
* {@inheritDoc}
*/
public function decrement(string $key, int $offset = 1): false|int
{
$key = static::validateKey($key, $this->prefix);

return apcu_dec($key, $offset);
}

/**
* {@inheritDoc}
*/
public function clean(): bool
{
return apcu_clear_cache();
}

/**
* {@inheritDoc}
*/
public function getCacheInfo(): array|false
{
return apcu_cache_info(true);
}

/**
* {@inheritDoc}
*/
public function getMetaData(string $key): ?array
{
$key = static::validateKey($key, $this->prefix);
$metadata = apcu_key_info($key);

if ($metadata !== null) {
return [
'expire' => $metadata['ttl'] > 0 ? Time::now()->getTimestamp() + $metadata['ttl'] : null,
'mtime' => $metadata['mtime'],
'data' => apcu_fetch($key),
];
}

return null;
}

/**
* {@inheritDoc}
*/
public function isSupported(): bool
{
return extension_loaded('apcu') && apcu_enabled();
}
}
Loading
Loading