diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index d261c5b..54f6615 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -11,12 +11,22 @@ jobs: include: - operating-system: 'ubuntu-latest' php-version: '8.1' + extensions: uv, eio - operating-system: 'ubuntu-latest' php-version: '8.2' + extensions: uv, eio - operating-system: 'ubuntu-latest' php-version: '8.3' + extensions: uv + style-fix: none + static-analysis: none + + - operating-system: 'ubuntu-latest' + php-version: '8.4' + extensions: uv + style-fix: none static-analysis: none - operating-system: 'windows-latest' @@ -26,7 +36,9 @@ jobs: - operating-system: 'macos-latest' php-version: '8.3' + extensions: uv job-description: 'on macOS' + style-fix: none static-analysis: none @@ -53,7 +65,7 @@ jobs: uses: shivammathur/setup-php@v2 with: php-version: ${{ matrix.php-version }} - extensions: eio-beta, uv-amphp/ext-uv@master + extensions: ${{ matrix.extensions }} - name: Get Composer cache directory id: composer-cache @@ -91,7 +103,7 @@ jobs: env: PHP_CS_FIXER_IGNORE_ENV: 1 run: vendor/bin/php-cs-fixer --diff --dry-run -v fix - if: runner.os != 'Windows' + if: runner.os != 'Windows' && matrix.style-fix != 'none' - name: Install composer-require-checker run: php -r 'file_put_contents("composer-require-checker.phar", file_get_contents("https://github.com/maglnet/ComposerRequireChecker/releases/download/3.7.0/composer-require-checker.phar"));' diff --git a/README.md b/README.md index 3d22bbe..b7f96c7 100644 --- a/README.md +++ b/README.md @@ -14,6 +14,10 @@ This package can be installed as a [Composer](https://getcomposer.org/) dependen composer require amphp/file ``` +## Requirements + +- PHP 8.1+ + `amphp/file` works out of the box without any PHP extensions. It uses multiple processes by default, but also comes with a blocking driver that uses PHP's blocking functions in the current process. @@ -163,7 +167,7 @@ array(13) { ## Security -If you discover any security related issues, please email [`me@kelunik.com`](mailto:me@kelunik.com) instead of using the issue tracker. +If you discover any security related issues, please use the private security issue reporter instead of using the public issue tracker. ## License diff --git a/composer-require-check.json b/composer-require-check.json index 93a5db8..3067f87 100644 --- a/composer-require-check.json +++ b/composer-require-check.json @@ -62,6 +62,7 @@ "EIO_S_IWUSR", "EIO_S_IXUSR", "UV", + "UVLoop", "uv_fs_chmod", "uv_fs_chown", "uv_fs_fstat", diff --git a/composer.json b/composer.json index 80dea86..6f46377 100644 --- a/composer.json +++ b/composer.json @@ -43,7 +43,7 @@ "require-dev": { "amphp/phpunit-util": "^3", "phpunit/phpunit": "^9", - "psalm/phar": "^5.4", + "psalm/phar": "5.22.2", "amphp/php-cs-fixer-config": "^2" }, "suggest": { @@ -59,11 +59,13 @@ "autoload-dev": { "psr-4": { "Amp\\File\\Test\\": "test", + "Amp\\Cache\\Test\\": "vendor/amphp/cache/test", "Amp\\Sync\\": "vendor/amphp/sync/test" } }, "config": { "preferred-install": { + "amphp/cache": "source", "amphp/sync": "source" } }, diff --git a/psalm.xml b/psalm.xml index e774aea..cf504b6 100644 --- a/psalm.xml +++ b/psalm.xml @@ -48,5 +48,12 @@ + + + + + + + diff --git a/src/Driver/ParallelFilesystemDriver.php b/src/Driver/ParallelFilesystemDriver.php index 4c978bf..f18a0a8 100644 --- a/src/Driver/ParallelFilesystemDriver.php +++ b/src/Driver/ParallelFilesystemDriver.php @@ -22,7 +22,7 @@ final class ParallelFilesystemDriver implements FilesystemDriver /** @var int Maximum number of workers to use for open files. */ private int $workerLimit; - /** @var \SplObjectStorage Worker storage. */ + /** @var \SplObjectStorage Worker storage. */ private \SplObjectStorage $workerStorage; /** @var Future Pending worker request */ @@ -31,11 +31,11 @@ final class ParallelFilesystemDriver implements FilesystemDriver /** * @param int $workerLimit Maximum number of workers to use from the pool for open files. */ - public function __construct(WorkerPool $pool = null, int $workerLimit = self::DEFAULT_WORKER_LIMIT) + public function __construct(?WorkerPool $pool = null, int $workerLimit = self::DEFAULT_WORKER_LIMIT) { $this->pool = $pool ?? workerPool(); $this->workerLimit = $workerLimit; - $this->workerStorage = new \SplObjectStorage; + $this->workerStorage = new \SplObjectStorage(); $this->pendingWorker = Future::complete(); } @@ -45,8 +45,11 @@ public function openFile(string $path, string $mode): ParallelFile $workerStorage = $this->workerStorage; $worker = new Internal\FileWorker($worker, static function (Worker $worker) use ($workerStorage): void { - \assert($workerStorage->contains($worker)); - if (($workerStorage[$worker] -=1) === 0 || !$worker->isRunning()) { + if (!$workerStorage->contains($worker)) { + return; + } + + if (($workerStorage[$worker] -= 1) === 0 || !$worker->isRunning()) { $workerStorage->detach($worker); } }); diff --git a/src/Driver/UvFile.php b/src/Driver/UvFile.php index 23765bc..5b3f770 100644 --- a/src/Driver/UvFile.php +++ b/src/Driver/UvFile.php @@ -1,5 +1,4 @@ eventLoopHandle = $driver->getHandle(); $this->onClose = new DeferredFuture; - - $this->priorVersion = \version_compare(\phpversion('uv'), '0.3.0', '<'); } public function read(?Cancellation $cancellation = null, int $length = self::DEFAULT_READ_LENGTH): ?string @@ -86,16 +79,6 @@ public function read(?Cancellation $cancellation = null, int $length = self::DEF $deferred->complete($length ? $buffer : null); }; - if ($this->priorVersion) { - $onRead = static function ($fh, $result, $buffer) use ($onRead): void { - if ($result < 0) { - $buffer = $result; // php-uv v0.3.0 changed the callback to put an int in $buffer on error. - } - - $onRead($result, $buffer); - }; - } - \uv_fs_read($this->eventLoopHandle, $this->fh, $this->position, $length, $onRead); $id = $cancellation?->subscribe(function (\Throwable $exception) use ($deferred): void { diff --git a/src/Driver/UvFilesystemDriver.php b/src/Driver/UvFilesystemDriver.php index 695bdc3..f396c70 100644 --- a/src/Driver/UvFilesystemDriver.php +++ b/src/Driver/UvFilesystemDriver.php @@ -1,5 +1,4 @@ =') && $driver->getHandle() instanceof \UVLoop; } - /** @var \UVLoop|resource Loop resource of type uv_loop or instance of \UVLoop. */ - private $eventLoopHandle; + private readonly \UVLoop $eventLoopHandle; private readonly Internal\UvPoll $poll; - /** @var bool True if ext-uv version is < 0.3.0. */ - private readonly bool $priorVersion; - - public function __construct(private readonly UvLoopDriver $driver) + public function __construct(private readonly EventLoopDriver $driver) { + if (!self::isSupported($driver)) { + throw new \Error('Event loop did not return a compatible handle'); + } + /** @psalm-suppress PropertyTypeCoercion */ $this->eventLoopHandle = $driver->getHandle(); $this->poll = new Internal\UvPoll($driver); - $this->priorVersion = \version_compare(\phpversion('uv'), '0.3.0', '<'); } public function openFile(string $path, string $mode): UvFile @@ -83,16 +85,6 @@ public function getStatus(string $path): ?array $deferred->complete($stat); }; - if ($this->priorVersion) { - $callback = static function ($fh, $stat) use ($callback): void { - if (empty($fh)) { - $stat = 0; - } - - $callback($stat); - }; - } - \uv_fs_stat($this->eventLoopHandle, $path, $callback); try { @@ -107,17 +99,9 @@ public function getLinkStatus(string $path): ?array $deferred = new DeferredFuture; $this->poll->listen(); - if ($this->priorVersion) { - $callback = static function ($fh, $stat) use ($deferred): void { - $deferred->complete(empty($fh) ? null : $stat); - }; - } else { - $callback = static function ($stat) use ($deferred): void { - $deferred->complete(\is_int($stat) ? null : $stat); - }; - } - - \uv_fs_lstat($this->eventLoopHandle, $path, $callback); + \uv_fs_lstat($this->eventLoopHandle, $path, static function ($stat) use ($deferred): void { + $deferred->complete(\is_int($stat) ? null : $stat); + }); try { return $deferred->getFuture()->await(); @@ -160,27 +144,14 @@ public function resolveSymlink(string $target): string $deferred = new DeferredFuture; $this->poll->listen(); - if ($this->priorVersion) { - $callback = static function ($fh, $target) use ($deferred): void { - if (!(bool) $fh) { - $deferred->error(new FilesystemException("Could not read symbolic link")); - return; - } - - $deferred->complete($target); - }; - } else { - $callback = static function ($target) use ($deferred): void { - if (\is_int($target)) { - $deferred->error(new FilesystemException("Could not read symbolic link")); - return; - } - - $deferred->complete($target); - }; - } + \uv_fs_readlink($this->eventLoopHandle, $target, static function ($target) use ($deferred): void { + if (\is_int($target)) { + $deferred->error(new FilesystemException("Could not read symbolic link")); + return; + } - \uv_fs_readlink($this->eventLoopHandle, $target, $callback); + $deferred->complete($target); + }); try { return $deferred->getFuture()->await(); @@ -297,28 +268,16 @@ public function listFiles(string $path): array $deferred = new DeferredFuture; $this->poll->listen(); - if ($this->priorVersion) { - \uv_fs_readdir($this->eventLoopHandle, $path, 0, static function ($fh, $data) use ($deferred, $path): void { - if (empty($fh) && $data !== 0) { - $deferred->error(new FilesystemException("Failed reading contents from {$path}")); - } elseif ($data === 0) { - $deferred->complete([]); - } else { - $deferred->complete($data); - } - }); - } else { - /** @noinspection PhpUndefinedFunctionInspection */ - \uv_fs_scandir($this->eventLoopHandle, $path, static function ($data) use ($deferred, $path): void { - if (\is_int($data) && $data !== 0) { - $deferred->error(new FilesystemException("Failed reading contents from {$path}")); - } elseif ($data === 0) { - $deferred->complete([]); - } else { - $deferred->complete($data); - } - }); - } + /** @noinspection PhpUndefinedFunctionInspection */ + \uv_fs_scandir($this->eventLoopHandle, $path, static function ($data) use ($deferred, $path): void { + if (\is_int($data) && $data !== 0) { + $deferred->error(new FilesystemException("Failed reading contents from {$path}")); + } elseif ($data === 0) { + $deferred->complete([]); + } else { + $deferred->complete($data); + } + }); try { return $deferred->getFuture()->await(); @@ -528,42 +487,24 @@ private function doFsRead(mixed $fileHandle, int $length): ?string { $deferred = new DeferredFuture; - if ($this->priorVersion) { - $callback = static function ($fileHandle, $readBytes, $buffer) use ($deferred): void { - $deferred->complete($readBytes < 0 ? null : $buffer); - }; - } else { - $callback = static function ($readBytes, $buffer) use ($deferred): void { - $deferred->complete($readBytes < 0 ? null : $buffer); - }; - } + $callback = static function ($readBytes, $buffer) use ($deferred): void { + $deferred->complete($readBytes < 0 ? null : $buffer); + }; \uv_fs_read($this->eventLoopHandle, $fileHandle, 0, $length, $callback); return $deferred->getFuture()->await(); } - private function doWrite(string $path, string $contents): void - { - } - private function createGenericCallback(DeferredFuture $deferred, string $error): \Closure { - $callback = static function (int $result) use ($deferred, $error): void { + return static function (int $result) use ($deferred, $error): void { if ($result !== 0) { $deferred->error(new FilesystemException($error)); return; } - $deferred->complete(null); + $deferred->complete(); }; - - if ($this->priorVersion) { - $callback = static function (bool $result) use ($callback): void { - $callback($result ? 0 : -1); - }; - } - - return $callback; } } diff --git a/src/FileCache.php b/src/FileCache.php new file mode 100644 index 0000000..f0971f5 --- /dev/null +++ b/src/FileCache.php @@ -0,0 +1,181 @@ +filesystem = $filesystem; + $this->directory = $directory = \rtrim($directory, "/\\"); + + $gcWatcher = static function () use ($directory, $mutex, $filesystem): void { + try { + $files = $filesystem->listFiles($directory); + + foreach ($files as $file) { + if (\strlen($file) !== 70 || !\str_ends_with($file, '.cache')) { + continue; + } + + try { + $lock = $mutex->acquire($file); + } catch (\Throwable) { + continue; + } + + try { + $handle = $filesystem->openFile($directory . '/' . $file, 'r'); + $ttl = $handle->read(length: 4); + + if ($ttl === null || \strlen($ttl) !== 4) { + $handle->close(); + continue; + } + + $ttl = \unpack('Nttl', $ttl)['ttl']; + if ($ttl < \time()) { + $filesystem->deleteFile($directory . '/' . $file); + } + } catch (\Throwable) { + // ignore + } finally { + $lock->release(); + } + } + } catch (\Throwable) { + // ignore + } + }; + + // trigger once, so short running scripts also GC and don't grow forever + EventLoop::defer($gcWatcher); + + $this->gcWatcher = EventLoop::repeat(300, $gcWatcher); + + EventLoop::unreference($this->gcWatcher); + } + + public function __destruct() + { + if ($this->gcWatcher !== null) { + EventLoop::cancel($this->gcWatcher); + } + } + + public function get(string $key): ?string + { + $filename = $this->getFilename($key); + + $lock = $this->lock($filename); + + try { + $cacheContent = $this->filesystem->read($this->directory . '/' . $filename); + + if (\strlen($cacheContent) < 4) { + return null; + } + + $ttl = \unpack('Nttl', \substr($cacheContent, 0, 4))['ttl']; + if ($ttl < \time()) { + $this->filesystem->deleteFile($this->directory . '/' . $filename); + + return null; + } + + $value = \substr($cacheContent, 4); + + \assert(\is_string($value)); + + return $value; + } catch (\Throwable) { + return null; + } finally { + $lock->release(); + } + } + + public function set(string $key, string $value, ?int $ttl = null): void + { + if ($ttl < 0) { + throw new \Error("Invalid cache TTL ({$ttl}); integer >= 0 or null required"); + } + + $filename = $this->getFilename($key); + + $lock = $this->lock($filename); + + if ($ttl === null) { + $ttl = \PHP_INT_MAX; + } else { + $ttl = \time() + $ttl; + } + + $encodedTtl = \pack('N', $ttl); + + try { + $this->filesystem->write($this->directory . '/' . $filename, $encodedTtl . $value); + } finally { + $lock->release(); + } + } + + public function delete(string $key): ?bool + { + $filename = $this->getFilename($key); + + $lock = $this->lock($filename); + + try { + $this->filesystem->deleteFile($this->directory . '/' . $filename); + } catch (FilesystemException) { + return false; + } finally { + $lock->release(); + } + + return true; + } + + private static function getFilename(string $key): string + { + return \hash('sha256', $key) . '.cache'; + } + + private function lock(string $key): Lock + { + try { + return $this->mutex->acquire($key); + } catch (\Throwable $exception) { + throw new CacheException( + \sprintf('Exception thrown when obtaining the lock for key "%s"', $key), + 0, + $exception + ); + } + } +} diff --git a/src/FileMutex.php b/src/FileMutex.php index 444dd68..f4fc820 100644 --- a/src/FileMutex.php +++ b/src/FileMutex.php @@ -2,33 +2,45 @@ namespace Amp\File; +use Amp\Cancellation; use Amp\Sync\Lock; use Amp\Sync\Mutex; +use Amp\Sync\SyncException; use function Amp\delay; final class FileMutex implements Mutex { private const LATENCY_TIMEOUT = 0.01; + private const DELAY_LIMIT = 1; + + private readonly Filesystem $filesystem; + + private readonly string $directory; /** * @param string $fileName Name of temporary file to use as a mutex. */ - public function __construct(private readonly string $fileName) + public function __construct(private readonly string $fileName, ?Filesystem $filesystem = null) { + $this->filesystem = $filesystem ?? filesystem(); + $this->directory = \dirname($this->fileName); } - public function acquire(): Lock + public function acquire(?Cancellation $cancellation = null): Lock { + if (!$this->filesystem->isDirectory($this->directory)) { + throw new SyncException(\sprintf('Directory of "%s" does not exist or is not a directory', $this->fileName)); + } $f = \fopen($this->fileName, 'c'); - while (true) { + + // Try to create the lock file. If the file already exists, someone else + // has the lock, so set an asynchronous timer and try again. + for ($attempt = 0; true; ++$attempt) { if (\flock($f, LOCK_EX|LOCK_NB)) { - // Return a lock object that can be used to release the lock on the mutex. $lock = new Lock(fn () => \flock($f, LOCK_UN)); - return $lock; } - - delay(self::LATENCY_TIMEOUT); + delay(\min(self::DELAY_LIMIT, self::LATENCY_TIMEOUT * (2 ** $attempt)), cancellation: $cancellation); } } } diff --git a/src/FilesystemException.php b/src/FilesystemException.php index d2b2cdb..5e58ea4 100644 --- a/src/FilesystemException.php +++ b/src/FilesystemException.php @@ -4,7 +4,7 @@ class FilesystemException extends \Exception { - public function __construct(string $message, \Throwable $previous = null) + public function __construct(string $message, ?\Throwable $previous = null) { parent::__construct($message, 0, $previous); } diff --git a/src/Internal/QueuedWritesFile.php b/src/Internal/QueuedWritesFile.php index ad91e2b..30b4d97 100644 --- a/src/Internal/QueuedWritesFile.php +++ b/src/Internal/QueuedWritesFile.php @@ -38,8 +38,8 @@ public function __construct( } $this->queue = new \SplQueue(); - $this->writable = $this->mode[0] !== 'r'; - $this->position = $this->mode[0] === 'a' ? $this->size : 0; + $this->writable = !\str_contains($this->mode, 'r') || \str_contains($this->mode, '+'); + $this->position = \str_contains($this->mode, 'a') ? $this->size : 0; } public function __destruct() diff --git a/src/Internal/UvPoll.php b/src/Internal/UvPoll.php index 5bccadf..fec197d 100644 --- a/src/Internal/UvPoll.php +++ b/src/Internal/UvPoll.php @@ -2,7 +2,7 @@ namespace Amp\File\Internal; -use Revolt\EventLoop\Driver\UvDriver as UvLoopDriver; +use Revolt\EventLoop\Driver as EventLoopDriver; /** @internal */ final class UvPoll @@ -11,7 +11,7 @@ final class UvPoll private int $requests = 0; - public function __construct(private readonly UvLoopDriver $driver) + public function __construct(private readonly EventLoopDriver $driver) { // Create dummy watcher to keep loop running while polling. diff --git a/src/KeyedFileMutex.php b/src/KeyedFileMutex.php new file mode 100644 index 0000000..a6a340c --- /dev/null +++ b/src/KeyedFileMutex.php @@ -0,0 +1,76 @@ +filesystem = $filesystem ?? filesystem(); + $this->directory = \rtrim($directory, "/\\"); + } + + public function acquire(string $key, ?Cancellation $cancellation = null): Lock + { + if (!$this->filesystem->isDirectory($this->directory)) { + throw new SyncException(\sprintf('Directory "%s" does not exist or is not a directory', $this->directory)); + } + + $filename = $this->getFilename($key); + + // Try to create the lock file. If the file already exists, someone else + // has the lock, so set an asynchronous timer and try again. + for ($attempt = 0; true; ++$attempt) { + try { + $file = $this->filesystem->openFile($filename, 'x'); + + // Return a lock object that can be used to release the lock on the mutex. + $lock = new Lock(fn () => $this->release($filename)); + + $file->close(); + + return $lock; + } catch (FilesystemException) { + delay(\min(self::DELAY_LIMIT, self::LATENCY_TIMEOUT * (2 ** $attempt)), cancellation: $cancellation); + } + } + } + + /** + * Releases the lock on the mutex. + * + * @throws SyncException + */ + private function release(string $filename): void + { + try { + $this->filesystem->deleteFile($filename); + } catch (\Throwable $exception) { + throw new SyncException( + 'Failed to unlock the mutex file: ' . $filename, + previous: $exception, + ); + } + } + + private function getFilename(string $key): string + { + return $this->directory . '/' . \hash('sha256', $key) . '.lock'; + } +} diff --git a/src/Whence.php b/src/Whence.php index e21b8cb..21ef1d0 100644 --- a/src/Whence.php +++ b/src/Whence.php @@ -8,10 +8,12 @@ enum Whence * Set position equal to offset bytes. */ case Start; + /** * Set position to current location plus offset. */ case Current; + /** * Set position to end-of-file plus offset. */ diff --git a/src/functions.php b/src/functions.php index c81dd7e..80f8b3d 100644 --- a/src/functions.php +++ b/src/functions.php @@ -55,7 +55,6 @@ function createDefaultDriver(): FilesystemDriver $driver = EventLoop::getDriver(); if (UvFilesystemDriver::isSupported($driver)) { - /** @var EventLoop\Driver\UvDriver $driver */ return new UvFilesystemDriver($driver); } diff --git a/test/FileCacheTest.php b/test/FileCacheTest.php new file mode 100644 index 0000000..ed3daa6 --- /dev/null +++ b/test/FileCacheTest.php @@ -0,0 +1,27 @@ +