diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml new file mode 100644 index 0000000..b270598 --- /dev/null +++ b/.github/workflows/ci.yml @@ -0,0 +1,36 @@ +name: CI +on: [push] +jobs: + Linter: + runs-on: ubuntu-latest + container: php:8.2 + steps: + - uses: actions/checkout@v3 + - name: Install composer + run: apt-get update -yq && apt-get install git wget procps unzip -y && pecl install -o -f redis && rm -rf /tmp/pear && docker-php-ext-enable redis && wget https://getcomposer.org/composer.phar && php composer.phar install --dev + - name: Validate composer.json and composer.lock + run: php composer.phar validate --strict + - name: Install dependencies + run: php composer.phar install --dev --prefer-dist --no-progress + - name: Run PHPCS Linter + run: php -d memory_limit=256M vendor/bin/phpcs -s --standard=ruleset.xml + PHPTest: + runs-on: ubuntu-latest + container: php:${{ matrix.php_version }} + strategy: + matrix: + php_version: [8.1, 8.2,8.3,8.4] + services: + redis: + image: redis:7.0 + options: >- + --health-cmd "redis-cli ping" + --health-interval 10s + --health-timeout 5s + --health-retries 5 + steps: + - uses: actions/checkout@v3 + - name: Install composer + run: apt-get update -yq && apt-get install git wget procps unzip -y && pecl install -o -f redis && rm -rf /tmp/pear && docker-php-ext-enable redis && wget https://getcomposer.org/composer.phar && php composer.phar install --dev + - name: Run PHP ${{ matrix.php_version }}} Unit Tests + run: php vendor/bin/phpunit --verbose --configuration phpunit.xml diff --git a/.gitignore b/.gitignore index 924bb34..aa30d8b 100644 --- a/.gitignore +++ b/.gitignore @@ -1,3 +1,2 @@ vendor/ -phpunit.xml -*.swp \ No newline at end of file +.phpunit.result.cache \ No newline at end of file diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml index c7bb222..fe02d9b 100644 --- a/.gitlab-ci.yml +++ b/.gitlab-ci.yml @@ -4,68 +4,40 @@ # Install git (the php image doesn't have it) which is required by composer apt-get update -yq - apt-get install git wget procps -y + apt-get install git wget procps unzip -y # Install pcntl and redis extentions pecl install -o -f redis \ - && rm -rf /tmp/pear \ - && docker-php-ext-enable redis + && rm -rf /tmp/pear \ + && docker-php-ext-enable redis docker-php-ext-install pcntl - # Hack to fix mismatched php versions - rm composer.lock - # Install Composer wget https://getcomposer.org/composer.phar - php composer.phar install + php composer.phar install --dev services: - - redis:latest + - redis:7 -# Test PHP 7.0 -test:7.0: - image: php:7.0 +# Test PHP +test: + image: php:$PHP_VERSION + parallel: + matrix: + - PHP_VERSION: [ "7.4", "8.0", "8.1", "8.2" ] before_script: - *docker_boostrap script: - - curl --location --output /usr/local/bin/phpunit https://phar.phpunit.de/phpunit-6.5.9.phar - - chmod +x /usr/local/bin/phpunit - - phpunit --verbose --configuration phpunit.xml.dist + - php vendor/bin/phpunit --verbose --configuration phpunit.xml tags: - docker -# Test PHP 7.1 -test:7.1: - image: php:7.1 - before_script: - - *docker_boostrap +# Codestandards +lint: + image: php:8.2 + allow_failure: true script: - - curl --location --output /usr/local/bin/phpunit https://phar.phpunit.de/phpunit-7.5.9.phar - - chmod +x /usr/local/bin/phpunit - - phpunit --verbose --configuration phpunit.xml.dist - tags: - - docker - -# Test PHP 7.2 -test:7.2: - image: php:7.2 - before_script: - - *docker_boostrap - script: - - curl --location --output /usr/local/bin/phpunit https://phar.phpunit.de/phpunit-7.5.9.phar - - chmod +x /usr/local/bin/phpunit - - phpunit --verbose --configuration phpunit.xml.dist - tags: - - docker - -# Test PHP 7.3 -test:7.3: - image: php:7.3 - before_script: - - *docker_boostrap - script: - - curl --location --output /usr/local/bin/phpunit https://phar.phpunit.de/phpunit.phar - - chmod +x /usr/local/bin/phpunit - - phpunit --verbose --configuration phpunit.xml.dist - tags: - - docker + - apt update && apt install -y wget unzip git + - wget https://getcomposer.org/composer.phar + - php composer.phar install --dev + - php -d memory_limit=256M vendor/bin/phpcs -s --standard=ruleset.xml \ No newline at end of file diff --git a/.travis.yml b/.travis.yml deleted file mode 100644 index 91c26ad..0000000 --- a/.travis.yml +++ /dev/null @@ -1,13 +0,0 @@ -language: php - -php: - - 7.0 - - 7.1 - - 7.2 - -services: - - redis-server - -before_script: - - echo "extension = redis.so" >> ~/.phpenv/versions/$(phpenv version-name)/etc/php.ini - - composer install diff --git a/CHANGELOG.md b/CHANGELOG.md index 90225ac..276f6e3 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,3 +1,71 @@ +# 2.5.3 (2025-06-08) +- Update typing of Log() to support all psr\log versions + +- # 2.5.2 (2025-06-08) +- Update typing of Log() to support all psr\log versions + +# 2.5.1 (2025-06-08) +- Update psr/log version requirements + +# 2.5.0 (2025-06-08) +- Update packages + +# 2.4.0 (2024-12-11) +- Update packages (psr/log ^3.0.2) + +# 2.3.0 (2024-09-04) +- Update packages + +# 2.2.0 (2023-03-20) +- Update pacakges +- Bump requirements to PHP >= 8.1 + +# 2.1.3 (2023-11-15) +- Resolved issue with SET EX TTL's using unix-timestamps + +# 2.1.2 (2023-03-22) +- Update composer packages +- Update git information (GitHub) + +# 2.1.1 (2023-03-20) +- Changed setex to set with EX values +- Added TTLs to missing keys + +## 2.1.0 (2023-02-07) +- Add PHP 8.1 / 8.2 unit tests +- Updated code to be PHP 8.2 compliant + +## 2.0.3 (2022-09-12) +- Update composer packages +- Added WoodpeckerCI tests +- Updated links in composer package +- Stricter typing + +## 2.0.2 (2022-02-15) +- Replace strftime with strtotime for PHP8.1 support +- Added processing class into proc line for easier debugging + +## 2.0.1 (2022-02-08) +- Fixed issue with lingering keys causing constant memory growth +- Add PHP8 support +- Composer upgrade + +## 2.0.0 (2021-02-19) +- Moved to PSR-4 +- Namespaced codebase +- Added more comments throughout + +## 1.4.7 (2020-04-11) +- Update PHPUnit to 9 +- Start adding return types + +## 1.4.6 (2020-01-10) +- Switched IF Statement order to prevent excess calls to redis. + +## 1.4.5 (2019-08-28) +- Added 'replaced' composer tag. +- Formatting changes. + ## 1.4.4 (2019-06-02) - Updated tests to run on GitLab CI. - Can now run tests locally using `gitlab-runner exec docker test:7.0` @@ -82,7 +150,7 @@ Changes by iskandar introduce improved support for using DSNs to connect to Redi * Pass queue name to afterEvent callback * Only declare RedisException if it doesn't already exist (Matt Heath) * Add support for Composer -* Fix missing and incorrect paths for Resque and Resque_Job_Status classes in demo (jjfrey) +* Fix missing and incorrect paths for Resque and \Resque\Job\Status classes in demo (jjfrey) * Disable autoload for the RedisException class_exists call (scragg0x) * General tidy up of comments and files/folders diff --git a/HOWITWORKS.md b/HOWITWORKS.md index ec85fa3..7168926 100644 --- a/HOWITWORKS.md +++ b/HOWITWORKS.md @@ -14,9 +14,9 @@ What happens when you call `Resque::enqueue()`? 4. `Resque_Job::create()` pushes the job to the requested queue (first argument) 5. `Resque_Job::create()`, if status monitoring is enabled for the job (fourth - argument), calls `Resque_Job_Status::create()` with the job ID as its only + argument), calls `\Resque\Job\Status::create()` with the job ID as its only argument -6. `Resque_Job_Status::create()` creates a key in Redis with the job ID in its +6. `\Resque\Job\Status::create()` creates a key in Redis with the job ID in its name, and the current status (as well as a couple of timestamps) as its value, then returns control to `Resque_Job::create()` 7. `Resque_Job::create()` returns control to `Resque::enqueue()`, with the job @@ -85,15 +85,15 @@ How do the workers process the queues? * Worker 1. The worker waits for the job process to complete 2. If the exit status is not 0, the worker calls `Resque_Job->fail()` with - a `Resque_Job_DirtyExitException` as its only argument. + a `Resque\Job\DirtyExitException` as its only argument. 3. `Resque_Job->fail()` triggers an `onFailure` event 4. `Resque_Job->fail()` updates the job status from `RUNNING` to `FAILED` 5. `Resque_Job->fail()` calls `Resque_Failure::create()` with the job - payload, the `Resque_Job_DirtyExitException`, the internal ID of the + payload, the `Resque\Job\DirtyExitException`, the internal ID of the worker, and the queue name as arguments 6. `Resque_Failure::create()` creates a new object of whatever type has been set as the `Resque_Failure` "backend" handler; by default, this is - a `Resque_Failure_Redis` object, whose constructor simply collects the + a `ResqueFailureRedis` object, whose constructor simply collects the data passed into `Resque_Failure::create()` and pushes it into Redis in the `failed` queue 7. `Resque_Job->fail()` increments two failure counters in Redis: one for diff --git a/README.md b/README.md index 79bc9a3..c2d7234 100644 --- a/README.md +++ b/README.md @@ -1,4 +1,4 @@ -php-resque: PHP Resque Worker (and Enqueue) [![Build Status](https://travis-ci.org/iDanoo/php-resque.svg?branch=master)](https://travis-ci.org/iDanoo/php-resque) +php-resque: PHP Background (Resque) Worker =========================================== Resque is a Redis-backed library for creating background jobs, placing @@ -42,7 +42,7 @@ On top of the original fork (chrisboulton/php-resque) I have added: ## Requirements ## -* PHP 7.0+ +* PHP 8.1+ * phpredis * Redis 2.2+ @@ -53,19 +53,9 @@ Composer package inside your project. If you're not familiar with Composer, please see . -1. Add php-resque to your application's composer.json. +1. Run `composer require idanoo/php-resque`. -```json -{ - "require": { - "idanoo/php-resque": "^1.4" - } -} -``` - -2. Run `composer install`. - -3. If you haven't already, add the Composer autoload to your project's +2. If you haven't already, add the Composer autoload to your project's initialization file. (example) ```sh @@ -80,11 +70,11 @@ Jobs are queued as follows: ```php // Required if redis is located elsewhere -Resque::setBackend('localhost:6379'); +Resque::setBackend('redis:6379'); $args = ['name' => 'TestName']; -Resque::enqueue('default', 'My_Job', $args); +Resque::enqueue('default', '\App\MyJobClass', $args); ``` ### Defining Jobs ### @@ -92,7 +82,9 @@ Resque::enqueue('default', 'My_Job', $args); Each job should be in its own class, and include a `perform` method. ```php -class My_Job +namespace \App; + +class MyJobClass { public function perform() { @@ -115,7 +107,9 @@ The `tearDown` method, if defined, will be called after the job finishes. ```php -class My_Job +namespace App; + +class MyJobClass { public function setUp() { @@ -139,17 +133,17 @@ class My_Job This method can be used to conveniently remove a job from a queue. ```php -// Removes job class 'My_Job' of queue 'default' -Resque::dequeue('default', ['My_Job']); +// Removes job class '\App\MyJobClass' of queue 'default' +Resque::dequeue('default', ['\App\MyJobClass']); -// Removes job class 'My_Job' with Job ID '087df5819a790ac666c9608e2234b21e' of queue 'default' -Resque::dequeue('default', ['My_Job' => '087df5819a790ac666c9608e2234b21e']); +// Removes job class '\App\MyJobClass' with Job ID '087df5819a790ac666c9608e2234b21e' of queue 'default' +Resque::dequeue('default', ['\App\MyJobClass' => '087df5819a790ac666c9608e2234b21e']); -// Removes job class 'My_Job' with arguments of queue 'default' -Resque::dequeue('default', ['My_Job' => ['foo' => 1, 'bar' => 2]]); +// Removes job class '\App\MyJobClass' with arguments of queue 'default' +Resque::dequeue('default', ['\App\MyJobClass' => ['foo' => 1, 'bar' => 2]]); // Removes multiple jobs -Resque::dequeue('default', ['My_Job', 'My_Job2']); +Resque::dequeue('default', ['\App\MyJobClass', '\App\MyJobClass2']); ``` If no jobs are given, this method will dequeue all jobs matching the provided queue. @@ -170,7 +164,7 @@ To track the status of a job, pass `true` as the fourth argument to returned: ```php -$token = Resque::enqueue('default', 'My_Job', $args, true); +$token = Resque::enqueue('default', '\App\MyJobClass', $args, true); echo $token; ``` @@ -367,8 +361,6 @@ your `APP_INCLUDE` script should initialize and register any listeners required for operation. If you have rolled your own worker manager, then it is again your responsibility to register listeners. -A sample plugin is included in the `extras` directory. - ### Events ### #### beforeFirstFork #### @@ -470,7 +462,7 @@ needing to directly examine the code), have a look at `HOWITWORKS.md`. ## Contributors ## ### Current Maintainers ### -* @iDanoo +* @idanoo ### Past Maintainer / Forked From ### diff --git a/bin/resque b/bin/resque index 23ff6d6..073f05a 100755 --- a/bin/resque +++ b/bin/resque @@ -9,7 +9,6 @@ $files = [ __DIR__ . '/../vendor/autoload.php', ]; -$found = false; foreach ($files as $file) { if (file_exists($file)) { require_once $file; @@ -25,6 +24,7 @@ if (!class_exists('Composer\Autoload\ClassLoader', false)) { ); } +// Set which queues to monitor '*' $QUEUE = getenv('QUEUE'); if (empty($QUEUE)) { die("Set QUEUE env var containing the list of queues to work.\n"); @@ -38,22 +38,26 @@ if (empty($QUEUE)) { */ $REDIS_BACKEND = getenv('REDIS_BACKEND'); -// A redis database number +/** + * REDIS_BACKEND_DB overrides default Redis DB + */ $REDIS_BACKEND_DB = getenv('REDIS_BACKEND_DB'); if (!empty($REDIS_BACKEND)) { if (empty($REDIS_BACKEND_DB)) { - Resque::setBackend($REDIS_BACKEND); + \Resque\Resque::setBackend($REDIS_BACKEND); } else { - Resque::setBackend($REDIS_BACKEND, $REDIS_BACKEND_DB); + \Resque\Resque::setBackend($REDIS_BACKEND, $REDIS_BACKEND_DB); } } +// Set Logging level $logLevel = false; $LOGGING = getenv('LOGLEVEL'); if (!empty($LOGGING)) { $logLevel = $LOGGING; } +// Bootstrap file $APP_INCLUDE = getenv('APP_INCLUDE'); if ($APP_INCLUDE) { if (!file_exists($APP_INCLUDE)) { @@ -66,41 +70,45 @@ if ($APP_INCLUDE) { // See if the APP_INCLUDE containes a logger object, // If none exists, fallback to internal logger if (!isset($logger) || !is_object($logger)) { - $logger = new Resque_Log($logLevel); + $logger = new \Resque\Log($logLevel); } +// Determines if blocking or not $BLOCKING = getenv('BLOCKING') !== FALSE; +// Interval to check for jobs $interval = 5; $INTERVAL = getenv('INTERVAL'); if (!empty($INTERVAL)) { $interval = $INTERVAL; } +// Sets worker count $count = 1; $COUNT = getenv('COUNT'); if (!empty($COUNT) && $COUNT > 1) { $count = $COUNT; } +// Determines redis key prefix $PREFIX = getenv('PREFIX'); if (!empty($PREFIX)) { - $logger->log(Psr\Log\LogLevel::INFO, 'Prefix set to {prefix}', ['prefix' => $PREFIX]); - Resque_Redis::prefix($PREFIX); + $logger->log(\Psr\Log\LogLevel::INFO, 'Prefix set to {prefix}', ['prefix' => $PREFIX]); + \Resque\Redis::prefix($PREFIX); } if ($count > 1) { for ($i = 0; $i < $count; ++$i) { - $pid = Resque::fork(); + $pid = \Resque\Resque::fork(); if ($pid === false || $pid === -1) { - $logger->log(Psr\Log\LogLevel::EMERGENCY, 'Could not fork worker {count}', ['count' => $i]); + $logger->log(\Psr\Log\LogLevel::EMERGENCY, 'Could not fork worker {count}', ['count' => $i]); die(); } elseif (!$pid) { // Child, start the worker $queues = explode(',', $QUEUE); - $worker = new Resque_Worker($queues); + $worker = new \Resque\Worker($queues); $worker->setLogger($logger); - $logger->log(Psr\Log\LogLevel::NOTICE, 'Starting worker {worker}', ['worker' => $worker]); + $logger->log(\Psr\Log\LogLevel::NOTICE, 'Starting worker {worker}', ['worker' => $worker]); $worker->work($interval, $BLOCKING); break; } @@ -108,7 +116,7 @@ if ($count > 1) { } else { // Start a single worker $queues = explode(',', $QUEUE); - $worker = new Resque_Worker($queues); + $worker = new \Resque\Worker($queues); $worker->setLogger($logger); $PIDFILE = getenv('PIDFILE'); @@ -117,6 +125,6 @@ if ($count > 1) { die('Could not write PID information to ' . $PIDFILE); } - $logger->log(Psr\Log\LogLevel::NOTICE, 'Starting worker {worker}', ['worker' => $worker]); + $logger->log(\Psr\Log\LogLevel::NOTICE, 'Starting worker {worker}', ['worker' => $worker]); $worker->work($interval, $BLOCKING); } diff --git a/build.xml b/build.xml deleted file mode 100644 index b51c486..0000000 --- a/build.xml +++ /dev/null @@ -1,18 +0,0 @@ - - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/composer.json b/composer.json index 85352af..15199f0 100644 --- a/composer.json +++ b/composer.json @@ -1,10 +1,13 @@ { "name": "idanoo/php-resque", - "version": "1.4.4", "type": "library", + "replace": { + "chrisboulton/php-resque": "*", + "danhunsaker/php-resque": "*" + }, "description": "Redis backed library for creating background jobs and processing them later. Based on resque for Ruby. Originally forked from chrisboulton/php-resque.", - "keywords": ["job", "background", "redis", "resque"], - "homepage": "http://www.github.com/idanoo/php-resque/", + "keywords": ["job", "background", "redis", "resque", "php"], + "homepage": "https://github.com/idanoo/php-resque", "license": "MIT", "authors": [ { @@ -13,25 +16,36 @@ } ], "require": { - "php": "^7.0", - "ext-pcntl": "*", - "ext-redis": "*", - "psr/log": "~1.0", - "colinmollenhour/credis": "^1.10" + "php": ">=8.1", + "psr/log": "^1.1 || ^2.0 || ^3.0", + "colinmollenhour/credis": "^1.14.0" }, "require-dev": { - "phpunit/phpunit": "^6" + "phpunit/phpunit": "^9", + "squizlabs/php_codesniffer": "3.*", + "phpcompatibility/php-compatibility": "^9.3", + "dealerdirect/phpcodesniffer-composer-installer": "^1.0" }, "bin": [ "bin/resque" ], "autoload": { - "psr-0": { - "Resque": "lib" + "psr-4": { + "Resque\\": "src/Resque" + } + }, + "autoload-dev": { + "psr-4": { + "Resque\\Test\\": "tests/Resque/Tests" } }, "support": { "issues": "https://github.com/idanoo/php-resque/issues", "source": "https://github.com/idanoo/php-resque" + }, + "config": { + "allow-plugins": { + "dealerdirect/phpcodesniffer-composer-installer": true + } } } diff --git a/composer.lock b/composer.lock index e89c818..bb2aed1 100644 --- a/composer.lock +++ b/composer.lock @@ -4,24 +4,27 @@ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "This file is @generated automatically" ], - "content-hash": "f7e4ae15d649e761fe2c5c3f6a466221", + "content-hash": "d8e5313006d5c73b54ee6a410b1ad016", "packages": [ { "name": "colinmollenhour/credis", - "version": "1.10.0", + "version": "v1.17.0", "source": { "type": "git", "url": "https://github.com/colinmollenhour/credis.git", - "reference": "8ab6db707c821055f9856b8cf76d5f44beb6fd8a" + "reference": "f4930b426f6b1238b687a1ffe6ee5af7f835b40a" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/colinmollenhour/credis/zipball/8ab6db707c821055f9856b8cf76d5f44beb6fd8a", - "reference": "8ab6db707c821055f9856b8cf76d5f44beb6fd8a", + "url": "https://api.github.com/repos/colinmollenhour/credis/zipball/f4930b426f6b1238b687a1ffe6ee5af7f835b40a", + "reference": "f4930b426f6b1238b687a1ffe6ee5af7f835b40a", "shasum": "" }, "require": { - "php": ">=5.4.0" + "php": ">=7.4.0" + }, + "suggest": { + "ext-redis": "Improved performance for communicating with redis" }, "type": "library", "autoload": { @@ -44,34 +47,38 @@ ], "description": "Credis is a lightweight interface to the Redis key-value store which wraps the phpredis library when available for better performance.", "homepage": "https://github.com/colinmollenhour/credis", - "time": "2018-05-07T14:45:04+00:00" + "support": { + "issues": "https://github.com/colinmollenhour/credis/issues", + "source": "https://github.com/colinmollenhour/credis/tree/v1.17.0" + }, + "time": "2025-02-10T18:58:46+00:00" }, { "name": "psr/log", - "version": "1.0.2", + "version": "3.0.2", "source": { "type": "git", "url": "https://github.com/php-fig/log.git", - "reference": "4ebe3a8bf773a19edfe0a84b6585ba3d401b724d" + "reference": "f16e1d5863e37f8d8c2a01719f5b34baa2b714d3" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/php-fig/log/zipball/4ebe3a8bf773a19edfe0a84b6585ba3d401b724d", - "reference": "4ebe3a8bf773a19edfe0a84b6585ba3d401b724d", + "url": "https://api.github.com/repos/php-fig/log/zipball/f16e1d5863e37f8d8c2a01719f5b34baa2b714d3", + "reference": "f16e1d5863e37f8d8c2a01719f5b34baa2b714d3", "shasum": "" }, "require": { - "php": ">=5.3.0" + "php": ">=8.0.0" }, "type": "library", "extra": { "branch-alias": { - "dev-master": "1.0.x-dev" + "dev-master": "3.x-dev" } }, "autoload": { "psr-4": { - "Psr\\Log\\": "Psr/Log/" + "Psr\\Log\\": "src" } }, "notification-url": "https://packagist.org/downloads/", @@ -81,7 +88,7 @@ "authors": [ { "name": "PHP-FIG", - "homepage": "http://www.php-fig.org/" + "homepage": "https://www.php-fig.org/" } ], "description": "Common interface for logging libraries", @@ -91,40 +98,119 @@ "psr", "psr-3" ], - "time": "2016-10-10T12:19:37+00:00" + "support": { + "source": "https://github.com/php-fig/log/tree/3.0.2" + }, + "time": "2024-09-11T13:17:53+00:00" } ], "packages-dev": [ { - "name": "doctrine/instantiator", - "version": "1.1.0", + "name": "dealerdirect/phpcodesniffer-composer-installer", + "version": "v1.0.0", "source": { "type": "git", - "url": "https://github.com/doctrine/instantiator.git", - "reference": "185b8868aa9bf7159f5f953ed5afb2d7fcdc3bda" + "url": "https://github.com/PHPCSStandards/composer-installer.git", + "reference": "4be43904336affa5c2f70744a348312336afd0da" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/doctrine/instantiator/zipball/185b8868aa9bf7159f5f953ed5afb2d7fcdc3bda", - "reference": "185b8868aa9bf7159f5f953ed5afb2d7fcdc3bda", + "url": "https://api.github.com/repos/PHPCSStandards/composer-installer/zipball/4be43904336affa5c2f70744a348312336afd0da", + "reference": "4be43904336affa5c2f70744a348312336afd0da", "shasum": "" }, "require": { - "php": "^7.1" + "composer-plugin-api": "^1.0 || ^2.0", + "php": ">=5.4", + "squizlabs/php_codesniffer": "^2.0 || ^3.1.0 || ^4.0" }, "require-dev": { - "athletic/athletic": "~0.1.8", - "ext-pdo": "*", - "ext-phar": "*", - "phpunit/phpunit": "^6.2.3", - "squizlabs/php_codesniffer": "^3.0.2" + "composer/composer": "*", + "ext-json": "*", + "ext-zip": "*", + "php-parallel-lint/php-parallel-lint": "^1.3.1", + "phpcompatibility/php-compatibility": "^9.0", + "yoast/phpunit-polyfills": "^1.0" }, - "type": "library", + "type": "composer-plugin", "extra": { - "branch-alias": { - "dev-master": "1.2.x-dev" + "class": "PHPCSStandards\\Composer\\Plugin\\Installers\\PHPCodeSniffer\\Plugin" + }, + "autoload": { + "psr-4": { + "PHPCSStandards\\Composer\\Plugin\\Installers\\PHPCodeSniffer\\": "src/" } }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Franck Nijhof", + "email": "franck.nijhof@dealerdirect.com", + "homepage": "http://www.frenck.nl", + "role": "Developer / IT Manager" + }, + { + "name": "Contributors", + "homepage": "https://github.com/PHPCSStandards/composer-installer/graphs/contributors" + } + ], + "description": "PHP_CodeSniffer Standards Composer Installer Plugin", + "homepage": "http://www.dealerdirect.com", + "keywords": [ + "PHPCodeSniffer", + "PHP_CodeSniffer", + "code quality", + "codesniffer", + "composer", + "installer", + "phpcbf", + "phpcs", + "plugin", + "qa", + "quality", + "standard", + "standards", + "style guide", + "stylecheck", + "tests" + ], + "support": { + "issues": "https://github.com/PHPCSStandards/composer-installer/issues", + "source": "https://github.com/PHPCSStandards/composer-installer" + }, + "time": "2023-01-05T11:28:13+00:00" + }, + { + "name": "doctrine/instantiator", + "version": "2.0.0", + "source": { + "type": "git", + "url": "https://github.com/doctrine/instantiator.git", + "reference": "c6222283fa3f4ac679f8b9ced9a4e23f163e80d0" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/doctrine/instantiator/zipball/c6222283fa3f4ac679f8b9ced9a4e23f163e80d0", + "reference": "c6222283fa3f4ac679f8b9ced9a4e23f163e80d0", + "shasum": "" + }, + "require": { + "php": "^8.1" + }, + "require-dev": { + "doctrine/coding-standard": "^11", + "ext-pdo": "*", + "ext-phar": "*", + "phpbench/phpbench": "^1.2", + "phpstan/phpstan": "^1.9.4", + "phpstan/phpstan-phpunit": "^1.3", + "phpunit/phpunit": "^9.5.27", + "vimeo/psalm": "^5.4" + }, + "type": "library", "autoload": { "psr-4": { "Doctrine\\Instantiator\\": "src/Doctrine/Instantiator/" @@ -138,50 +224,70 @@ { "name": "Marco Pivetta", "email": "ocramius@gmail.com", - "homepage": "http://ocramius.github.com/" + "homepage": "https://ocramius.github.io/" } ], "description": "A small, lightweight utility to instantiate objects in PHP without invoking their constructors", - "homepage": "https://github.com/doctrine/instantiator", + "homepage": "https://www.doctrine-project.org/projects/instantiator.html", "keywords": [ "constructor", "instantiate" ], - "time": "2017-07-22T11:58:36+00:00" + "support": { + "issues": "https://github.com/doctrine/instantiator/issues", + "source": "https://github.com/doctrine/instantiator/tree/2.0.0" + }, + "funding": [ + { + "url": "https://www.doctrine-project.org/sponsorship.html", + "type": "custom" + }, + { + "url": "https://www.patreon.com/phpdoctrine", + "type": "patreon" + }, + { + "url": "https://tidelift.com/funding/github/packagist/doctrine%2Finstantiator", + "type": "tidelift" + } + ], + "time": "2022-12-30T00:23:10+00:00" }, { "name": "myclabs/deep-copy", - "version": "1.8.1", + "version": "1.13.1", "source": { "type": "git", "url": "https://github.com/myclabs/DeepCopy.git", - "reference": "3e01bdad3e18354c3dce54466b7fbe33a9f9f7f8" + "reference": "1720ddd719e16cf0db4eb1c6eca108031636d46c" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/myclabs/DeepCopy/zipball/3e01bdad3e18354c3dce54466b7fbe33a9f9f7f8", - "reference": "3e01bdad3e18354c3dce54466b7fbe33a9f9f7f8", + "url": "https://api.github.com/repos/myclabs/DeepCopy/zipball/1720ddd719e16cf0db4eb1c6eca108031636d46c", + "reference": "1720ddd719e16cf0db4eb1c6eca108031636d46c", "shasum": "" }, "require": { - "php": "^7.1" + "php": "^7.1 || ^8.0" }, - "replace": { - "myclabs/deep-copy": "self.version" + "conflict": { + "doctrine/collections": "<1.6.8", + "doctrine/common": "<2.13.3 || >=3 <3.2.2" }, "require-dev": { - "doctrine/collections": "^1.0", - "doctrine/common": "^2.6", - "phpunit/phpunit": "^7.1" + "doctrine/collections": "^1.6.8", + "doctrine/common": "^2.13.3 || ^3.2.2", + "phpspec/prophecy": "^1.10", + "phpunit/phpunit": "^7.5.20 || ^8.5.23 || ^9.5.13" }, "type": "library", "autoload": { - "psr-4": { - "DeepCopy\\": "src/DeepCopy/" - }, "files": [ "src/DeepCopy/deep_copy.php" - ] + ], + "psr-4": { + "DeepCopy\\": "src/DeepCopy/" + } }, "notification-url": "https://packagist.org/downloads/", "license": [ @@ -195,32 +301,102 @@ "object", "object graph" ], - "time": "2018-06-11T23:09:50+00:00" + "support": { + "issues": "https://github.com/myclabs/DeepCopy/issues", + "source": "https://github.com/myclabs/DeepCopy/tree/1.13.1" + }, + "funding": [ + { + "url": "https://tidelift.com/funding/github/packagist/myclabs/deep-copy", + "type": "tidelift" + } + ], + "time": "2025-04-29T12:36:36+00:00" }, { - "name": "phar-io/manifest", - "version": "1.0.1", + "name": "nikic/php-parser", + "version": "v5.5.0", "source": { "type": "git", - "url": "https://github.com/phar-io/manifest.git", - "reference": "2df402786ab5368a0169091f61a7c1e0eb6852d0" + "url": "https://github.com/nikic/PHP-Parser.git", + "reference": "ae59794362fe85e051a58ad36b289443f57be7a9" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/phar-io/manifest/zipball/2df402786ab5368a0169091f61a7c1e0eb6852d0", - "reference": "2df402786ab5368a0169091f61a7c1e0eb6852d0", + "url": "https://api.github.com/repos/nikic/PHP-Parser/zipball/ae59794362fe85e051a58ad36b289443f57be7a9", + "reference": "ae59794362fe85e051a58ad36b289443f57be7a9", + "shasum": "" + }, + "require": { + "ext-ctype": "*", + "ext-json": "*", + "ext-tokenizer": "*", + "php": ">=7.4" + }, + "require-dev": { + "ircmaxell/php-yacc": "^0.0.7", + "phpunit/phpunit": "^9.0" + }, + "bin": [ + "bin/php-parse" + ], + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "5.0-dev" + } + }, + "autoload": { + "psr-4": { + "PhpParser\\": "lib/PhpParser" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Nikita Popov" + } + ], + "description": "A PHP parser written in PHP", + "keywords": [ + "parser", + "php" + ], + "support": { + "issues": "https://github.com/nikic/PHP-Parser/issues", + "source": "https://github.com/nikic/PHP-Parser/tree/v5.5.0" + }, + "time": "2025-05-31T08:24:38+00:00" + }, + { + "name": "phar-io/manifest", + "version": "2.0.4", + "source": { + "type": "git", + "url": "https://github.com/phar-io/manifest.git", + "reference": "54750ef60c58e43759730615a392c31c80e23176" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/phar-io/manifest/zipball/54750ef60c58e43759730615a392c31c80e23176", + "reference": "54750ef60c58e43759730615a392c31c80e23176", "shasum": "" }, "require": { "ext-dom": "*", + "ext-libxml": "*", "ext-phar": "*", - "phar-io/version": "^1.0.1", - "php": "^5.6 || ^7.0" + "ext-xmlwriter": "*", + "phar-io/version": "^3.0.1", + "php": "^7.2 || ^8.0" }, "type": "library", "extra": { "branch-alias": { - "dev-master": "1.0.x-dev" + "dev-master": "2.0.x-dev" } }, "autoload": { @@ -250,24 +426,34 @@ } ], "description": "Component for reading phar.io manifest information from a PHP Archive (PHAR)", - "time": "2017-03-05T18:14:27+00:00" + "support": { + "issues": "https://github.com/phar-io/manifest/issues", + "source": "https://github.com/phar-io/manifest/tree/2.0.4" + }, + "funding": [ + { + "url": "https://github.com/theseer", + "type": "github" + } + ], + "time": "2024-03-03T12:33:53+00:00" }, { "name": "phar-io/version", - "version": "1.0.1", + "version": "3.2.1", "source": { "type": "git", "url": "https://github.com/phar-io/version.git", - "reference": "a70c0ced4be299a63d32fa96d9281d03e94041df" + "reference": "4f7fd7836c6f332bb2933569e566a0d6c4cbed74" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/phar-io/version/zipball/a70c0ced4be299a63d32fa96d9281d03e94041df", - "reference": "a70c0ced4be299a63d32fa96d9281d03e94041df", + "url": "https://api.github.com/repos/phar-io/version/zipball/4f7fd7836c6f332bb2933569e566a0d6c4cbed74", + "reference": "4f7fd7836c6f332bb2933569e566a0d6c4cbed74", "shasum": "" }, "require": { - "php": "^5.6 || ^7.0" + "php": "^7.2 || ^8.0" }, "type": "library", "autoload": { @@ -297,259 +483,114 @@ } ], "description": "Library for handling version information and constraints", - "time": "2017-03-05T17:38:23+00:00" + "support": { + "issues": "https://github.com/phar-io/version/issues", + "source": "https://github.com/phar-io/version/tree/3.2.1" + }, + "time": "2022-02-21T01:04:05+00:00" }, { - "name": "phpdocumentor/reflection-common", - "version": "1.0.1", + "name": "phpcompatibility/php-compatibility", + "version": "9.3.5", "source": { "type": "git", - "url": "https://github.com/phpDocumentor/ReflectionCommon.git", - "reference": "21bdeb5f65d7ebf9f43b1b25d404f87deab5bfb6" + "url": "https://github.com/PHPCompatibility/PHPCompatibility.git", + "reference": "9fb324479acf6f39452e0655d2429cc0d3914243" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/phpDocumentor/ReflectionCommon/zipball/21bdeb5f65d7ebf9f43b1b25d404f87deab5bfb6", - "reference": "21bdeb5f65d7ebf9f43b1b25d404f87deab5bfb6", + "url": "https://api.github.com/repos/PHPCompatibility/PHPCompatibility/zipball/9fb324479acf6f39452e0655d2429cc0d3914243", + "reference": "9fb324479acf6f39452e0655d2429cc0d3914243", "shasum": "" }, "require": { - "php": ">=5.5" + "php": ">=5.3", + "squizlabs/php_codesniffer": "^2.3 || ^3.0.2" + }, + "conflict": { + "squizlabs/php_codesniffer": "2.6.2" }, "require-dev": { - "phpunit/phpunit": "^4.6" + "phpunit/phpunit": "~4.5 || ^5.0 || ^6.0 || ^7.0" }, - "type": "library", - "extra": { - "branch-alias": { - "dev-master": "1.0.x-dev" - } - }, - "autoload": { - "psr-4": { - "phpDocumentor\\Reflection\\": [ - "src" - ] - } + "suggest": { + "dealerdirect/phpcodesniffer-composer-installer": "^0.5 || This Composer plugin will sort out the PHPCS 'installed_paths' automatically.", + "roave/security-advisories": "dev-master || Helps prevent installing dependencies with known security issues." }, + "type": "phpcodesniffer-standard", "notification-url": "https://packagist.org/downloads/", "license": [ - "MIT" + "LGPL-3.0-or-later" ], "authors": [ { - "name": "Jaap van Otterdijk", - "email": "opensource@ijaap.nl" - } - ], - "description": "Common reflection classes used by phpdocumentor to reflect the code structure", - "homepage": "http://www.phpdoc.org", - "keywords": [ - "FQSEN", - "phpDocumentor", - "phpdoc", - "reflection", - "static analysis" - ], - "time": "2017-09-11T18:02:19+00:00" - }, - { - "name": "phpdocumentor/reflection-docblock", - "version": "4.3.0", - "source": { - "type": "git", - "url": "https://github.com/phpDocumentor/ReflectionDocBlock.git", - "reference": "94fd0001232e47129dd3504189fa1c7225010d08" - }, - "dist": { - "type": "zip", - "url": "https://api.github.com/repos/phpDocumentor/ReflectionDocBlock/zipball/94fd0001232e47129dd3504189fa1c7225010d08", - "reference": "94fd0001232e47129dd3504189fa1c7225010d08", - "shasum": "" - }, - "require": { - "php": "^7.0", - "phpdocumentor/reflection-common": "^1.0.0", - "phpdocumentor/type-resolver": "^0.4.0", - "webmozart/assert": "^1.0" - }, - "require-dev": { - "doctrine/instantiator": "~1.0.5", - "mockery/mockery": "^1.0", - "phpunit/phpunit": "^6.4" - }, - "type": "library", - "extra": { - "branch-alias": { - "dev-master": "4.x-dev" - } - }, - "autoload": { - "psr-4": { - "phpDocumentor\\Reflection\\": [ - "src/" - ] - } - }, - "notification-url": "https://packagist.org/downloads/", - "license": [ - "MIT" - ], - "authors": [ - { - "name": "Mike van Riel", - "email": "me@mikevanriel.com" - } - ], - "description": "With this component, a library can provide support for annotations via DocBlocks or otherwise retrieve information that is embedded in a DocBlock.", - "time": "2017-11-30T07:14:17+00:00" - }, - { - "name": "phpdocumentor/type-resolver", - "version": "0.4.0", - "source": { - "type": "git", - "url": "https://github.com/phpDocumentor/TypeResolver.git", - "reference": "9c977708995954784726e25d0cd1dddf4e65b0f7" - }, - "dist": { - "type": "zip", - "url": "https://api.github.com/repos/phpDocumentor/TypeResolver/zipball/9c977708995954784726e25d0cd1dddf4e65b0f7", - "reference": "9c977708995954784726e25d0cd1dddf4e65b0f7", - "shasum": "" - }, - "require": { - "php": "^5.5 || ^7.0", - "phpdocumentor/reflection-common": "^1.0" - }, - "require-dev": { - "mockery/mockery": "^0.9.4", - "phpunit/phpunit": "^5.2||^4.8.24" - }, - "type": "library", - "extra": { - "branch-alias": { - "dev-master": "1.0.x-dev" - } - }, - "autoload": { - "psr-4": { - "phpDocumentor\\Reflection\\": [ - "src/" - ] - } - }, - "notification-url": "https://packagist.org/downloads/", - "license": [ - "MIT" - ], - "authors": [ - { - "name": "Mike van Riel", - "email": "me@mikevanriel.com" - } - ], - "time": "2017-07-14T14:27:02+00:00" - }, - { - "name": "phpspec/prophecy", - "version": "1.8.0", - "source": { - "type": "git", - "url": "https://github.com/phpspec/prophecy.git", - "reference": "4ba436b55987b4bf311cb7c6ba82aa528aac0a06" - }, - "dist": { - "type": "zip", - "url": "https://api.github.com/repos/phpspec/prophecy/zipball/4ba436b55987b4bf311cb7c6ba82aa528aac0a06", - "reference": "4ba436b55987b4bf311cb7c6ba82aa528aac0a06", - "shasum": "" - }, - "require": { - "doctrine/instantiator": "^1.0.2", - "php": "^5.3|^7.0", - "phpdocumentor/reflection-docblock": "^2.0|^3.0.2|^4.0", - "sebastian/comparator": "^1.1|^2.0|^3.0", - "sebastian/recursion-context": "^1.0|^2.0|^3.0" - }, - "require-dev": { - "phpspec/phpspec": "^2.5|^3.2", - "phpunit/phpunit": "^4.8.35 || ^5.7 || ^6.5 || ^7.1" - }, - "type": "library", - "extra": { - "branch-alias": { - "dev-master": "1.8.x-dev" - } - }, - "autoload": { - "psr-0": { - "Prophecy\\": "src/" - } - }, - "notification-url": "https://packagist.org/downloads/", - "license": [ - "MIT" - ], - "authors": [ - { - "name": "Konstantin Kudryashov", - "email": "ever.zet@gmail.com", - "homepage": "http://everzet.com" + "name": "Wim Godden", + "homepage": "https://github.com/wimg", + "role": "lead" }, { - "name": "Marcello Duarte", - "email": "marcello.duarte@gmail.com" + "name": "Juliette Reinders Folmer", + "homepage": "https://github.com/jrfnl", + "role": "lead" + }, + { + "name": "Contributors", + "homepage": "https://github.com/PHPCompatibility/PHPCompatibility/graphs/contributors" } ], - "description": "Highly opinionated mocking framework for PHP 5.3+", - "homepage": "https://github.com/phpspec/prophecy", + "description": "A set of sniffs for PHP_CodeSniffer that checks for PHP cross-version compatibility.", + "homepage": "http://techblog.wimgodden.be/tag/codesniffer/", "keywords": [ - "Double", - "Dummy", - "fake", - "mock", - "spy", - "stub" + "compatibility", + "phpcs", + "standards" ], - "time": "2018-08-05T17:53:17+00:00" + "support": { + "issues": "https://github.com/PHPCompatibility/PHPCompatibility/issues", + "source": "https://github.com/PHPCompatibility/PHPCompatibility" + }, + "time": "2019-12-27T09:44:58+00:00" }, { "name": "phpunit/php-code-coverage", - "version": "5.3.2", + "version": "9.2.32", "source": { "type": "git", "url": "https://github.com/sebastianbergmann/php-code-coverage.git", - "reference": "c89677919c5dd6d3b3852f230a663118762218ac" + "reference": "85402a822d1ecf1db1096959413d35e1c37cf1a5" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/php-code-coverage/zipball/c89677919c5dd6d3b3852f230a663118762218ac", - "reference": "c89677919c5dd6d3b3852f230a663118762218ac", + "url": "https://api.github.com/repos/sebastianbergmann/php-code-coverage/zipball/85402a822d1ecf1db1096959413d35e1c37cf1a5", + "reference": "85402a822d1ecf1db1096959413d35e1c37cf1a5", "shasum": "" }, "require": { "ext-dom": "*", + "ext-libxml": "*", "ext-xmlwriter": "*", - "php": "^7.0", - "phpunit/php-file-iterator": "^1.4.2", - "phpunit/php-text-template": "^1.2.1", - "phpunit/php-token-stream": "^2.0.1", - "sebastian/code-unit-reverse-lookup": "^1.0.1", - "sebastian/environment": "^3.0", - "sebastian/version": "^2.0.1", - "theseer/tokenizer": "^1.1" + "nikic/php-parser": "^4.19.1 || ^5.1.0", + "php": ">=7.3", + "phpunit/php-file-iterator": "^3.0.6", + "phpunit/php-text-template": "^2.0.4", + "sebastian/code-unit-reverse-lookup": "^2.0.3", + "sebastian/complexity": "^2.0.3", + "sebastian/environment": "^5.1.5", + "sebastian/lines-of-code": "^1.0.4", + "sebastian/version": "^3.0.2", + "theseer/tokenizer": "^1.2.3" }, "require-dev": { - "phpunit/phpunit": "^6.0" + "phpunit/phpunit": "^9.6" }, "suggest": { - "ext-xdebug": "^2.5.5" + "ext-pcov": "PHP extension that provides line coverage", + "ext-xdebug": "PHP extension that provides line coverage as well as branch and path coverage" }, "type": "library", "extra": { "branch-alias": { - "dev-master": "5.3.x-dev" + "dev-main": "9.2.x-dev" } }, "autoload": { @@ -575,29 +616,43 @@ "testing", "xunit" ], - "time": "2018-04-06T15:36:58+00:00" + "support": { + "issues": "https://github.com/sebastianbergmann/php-code-coverage/issues", + "security": "https://github.com/sebastianbergmann/php-code-coverage/security/policy", + "source": "https://github.com/sebastianbergmann/php-code-coverage/tree/9.2.32" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2024-08-22T04:23:01+00:00" }, { "name": "phpunit/php-file-iterator", - "version": "1.4.5", + "version": "3.0.6", "source": { "type": "git", "url": "https://github.com/sebastianbergmann/php-file-iterator.git", - "reference": "730b01bc3e867237eaac355e06a36b85dd93a8b4" + "reference": "cf1c2e7c203ac650e352f4cc675a7021e7d1b3cf" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/php-file-iterator/zipball/730b01bc3e867237eaac355e06a36b85dd93a8b4", - "reference": "730b01bc3e867237eaac355e06a36b85dd93a8b4", + "url": "https://api.github.com/repos/sebastianbergmann/php-file-iterator/zipball/cf1c2e7c203ac650e352f4cc675a7021e7d1b3cf", + "reference": "cf1c2e7c203ac650e352f4cc675a7021e7d1b3cf", "shasum": "" }, "require": { - "php": ">=5.3.3" + "php": ">=7.3" + }, + "require-dev": { + "phpunit/phpunit": "^9.3" }, "type": "library", "extra": { "branch-alias": { - "dev-master": "1.4.x-dev" + "dev-master": "3.0-dev" } }, "autoload": { @@ -612,7 +667,7 @@ "authors": [ { "name": "Sebastian Bergmann", - "email": "sb@sebastian-bergmann.de", + "email": "sebastian@phpunit.de", "role": "lead" } ], @@ -622,26 +677,107 @@ "filesystem", "iterator" ], - "time": "2017-11-27T13:52:08+00:00" + "support": { + "issues": "https://github.com/sebastianbergmann/php-file-iterator/issues", + "source": "https://github.com/sebastianbergmann/php-file-iterator/tree/3.0.6" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2021-12-02T12:48:52+00:00" }, { - "name": "phpunit/php-text-template", - "version": "1.2.1", + "name": "phpunit/php-invoker", + "version": "3.1.1", "source": { "type": "git", - "url": "https://github.com/sebastianbergmann/php-text-template.git", - "reference": "31f8b717e51d9a2afca6c9f046f5d69fc27c8686" + "url": "https://github.com/sebastianbergmann/php-invoker.git", + "reference": "5a10147d0aaf65b58940a0b72f71c9ac0423cc67" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/php-text-template/zipball/31f8b717e51d9a2afca6c9f046f5d69fc27c8686", - "reference": "31f8b717e51d9a2afca6c9f046f5d69fc27c8686", + "url": "https://api.github.com/repos/sebastianbergmann/php-invoker/zipball/5a10147d0aaf65b58940a0b72f71c9ac0423cc67", + "reference": "5a10147d0aaf65b58940a0b72f71c9ac0423cc67", "shasum": "" }, "require": { - "php": ">=5.3.3" + "php": ">=7.3" + }, + "require-dev": { + "ext-pcntl": "*", + "phpunit/phpunit": "^9.3" + }, + "suggest": { + "ext-pcntl": "*" }, "type": "library", + "extra": { + "branch-alias": { + "dev-master": "3.1-dev" + } + }, + "autoload": { + "classmap": [ + "src/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de", + "role": "lead" + } + ], + "description": "Invoke callables with a timeout", + "homepage": "https://github.com/sebastianbergmann/php-invoker/", + "keywords": [ + "process" + ], + "support": { + "issues": "https://github.com/sebastianbergmann/php-invoker/issues", + "source": "https://github.com/sebastianbergmann/php-invoker/tree/3.1.1" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2020-09-28T05:58:55+00:00" + }, + { + "name": "phpunit/php-text-template", + "version": "2.0.4", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/php-text-template.git", + "reference": "5da5f67fc95621df9ff4c4e5a84d6a8a2acf7c28" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/php-text-template/zipball/5da5f67fc95621df9ff4c4e5a84d6a8a2acf7c28", + "reference": "5da5f67fc95621df9ff4c4e5a84d6a8a2acf7c28", + "shasum": "" + }, + "require": { + "php": ">=7.3" + }, + "require-dev": { + "phpunit/phpunit": "^9.3" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "2.0-dev" + } + }, "autoload": { "classmap": [ "src/" @@ -663,32 +799,42 @@ "keywords": [ "template" ], - "time": "2015-06-21T13:50:34+00:00" + "support": { + "issues": "https://github.com/sebastianbergmann/php-text-template/issues", + "source": "https://github.com/sebastianbergmann/php-text-template/tree/2.0.4" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2020-10-26T05:33:50+00:00" }, { "name": "phpunit/php-timer", - "version": "1.0.9", + "version": "5.0.3", "source": { "type": "git", "url": "https://github.com/sebastianbergmann/php-timer.git", - "reference": "3dcf38ca72b158baf0bc245e9184d3fdffa9c46f" + "reference": "5a63ce20ed1b5bf577850e2c4e87f4aa902afbd2" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/php-timer/zipball/3dcf38ca72b158baf0bc245e9184d3fdffa9c46f", - "reference": "3dcf38ca72b158baf0bc245e9184d3fdffa9c46f", + "url": "https://api.github.com/repos/sebastianbergmann/php-timer/zipball/5a63ce20ed1b5bf577850e2c4e87f4aa902afbd2", + "reference": "5a63ce20ed1b5bf577850e2c4e87f4aa902afbd2", "shasum": "" }, "require": { - "php": "^5.3.3 || ^7.0" + "php": ">=7.3" }, "require-dev": { - "phpunit/phpunit": "^4.8.35 || ^5.7 || ^6.0" + "phpunit/phpunit": "^9.3" }, "type": "library", "extra": { "branch-alias": { - "dev-master": "1.0-dev" + "dev-master": "5.0-dev" } }, "autoload": { @@ -703,7 +849,7 @@ "authors": [ { "name": "Sebastian Bergmann", - "email": "sb@sebastian-bergmann.de", + "email": "sebastian@phpunit.de", "role": "lead" } ], @@ -712,106 +858,64 @@ "keywords": [ "timer" ], - "time": "2017-02-26T11:10:40+00:00" - }, - { - "name": "phpunit/php-token-stream", - "version": "2.0.2", - "source": { - "type": "git", - "url": "https://github.com/sebastianbergmann/php-token-stream.git", - "reference": "791198a2c6254db10131eecfe8c06670700904db" + "support": { + "issues": "https://github.com/sebastianbergmann/php-timer/issues", + "source": "https://github.com/sebastianbergmann/php-timer/tree/5.0.3" }, - "dist": { - "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/php-token-stream/zipball/791198a2c6254db10131eecfe8c06670700904db", - "reference": "791198a2c6254db10131eecfe8c06670700904db", - "shasum": "" - }, - "require": { - "ext-tokenizer": "*", - "php": "^7.0" - }, - "require-dev": { - "phpunit/phpunit": "^6.2.4" - }, - "type": "library", - "extra": { - "branch-alias": { - "dev-master": "2.0-dev" - } - }, - "autoload": { - "classmap": [ - "src/" - ] - }, - "notification-url": "https://packagist.org/downloads/", - "license": [ - "BSD-3-Clause" - ], - "authors": [ + "funding": [ { - "name": "Sebastian Bergmann", - "email": "sebastian@phpunit.de" + "url": "https://github.com/sebastianbergmann", + "type": "github" } ], - "description": "Wrapper around PHP's tokenizer extension.", - "homepage": "https://github.com/sebastianbergmann/php-token-stream/", - "keywords": [ - "tokenizer" - ], - "time": "2017-11-27T05:48:46+00:00" + "time": "2020-10-26T13:16:10+00:00" }, { "name": "phpunit/phpunit", - "version": "6.5.13", + "version": "9.6.23", "source": { "type": "git", "url": "https://github.com/sebastianbergmann/phpunit.git", - "reference": "0973426fb012359b2f18d3bd1e90ef1172839693" + "reference": "43d2cb18d0675c38bd44982a5d1d88f6d53d8d95" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/phpunit/zipball/0973426fb012359b2f18d3bd1e90ef1172839693", - "reference": "0973426fb012359b2f18d3bd1e90ef1172839693", + "url": "https://api.github.com/repos/sebastianbergmann/phpunit/zipball/43d2cb18d0675c38bd44982a5d1d88f6d53d8d95", + "reference": "43d2cb18d0675c38bd44982a5d1d88f6d53d8d95", "shasum": "" }, "require": { + "doctrine/instantiator": "^1.5.0 || ^2", "ext-dom": "*", "ext-json": "*", "ext-libxml": "*", "ext-mbstring": "*", "ext-xml": "*", - "myclabs/deep-copy": "^1.6.1", - "phar-io/manifest": "^1.0.1", - "phar-io/version": "^1.0", - "php": "^7.0", - "phpspec/prophecy": "^1.7", - "phpunit/php-code-coverage": "^5.3", - "phpunit/php-file-iterator": "^1.4.3", - "phpunit/php-text-template": "^1.2.1", - "phpunit/php-timer": "^1.0.9", - "phpunit/phpunit-mock-objects": "^5.0.9", - "sebastian/comparator": "^2.1", - "sebastian/diff": "^2.0", - "sebastian/environment": "^3.1", - "sebastian/exporter": "^3.1", - "sebastian/global-state": "^2.0", - "sebastian/object-enumerator": "^3.0.3", - "sebastian/resource-operations": "^1.0", - "sebastian/version": "^2.0.1" - }, - "conflict": { - "phpdocumentor/reflection-docblock": "3.0.2", - "phpunit/dbunit": "<3.0" - }, - "require-dev": { - "ext-pdo": "*" + "ext-xmlwriter": "*", + "myclabs/deep-copy": "^1.13.1", + "phar-io/manifest": "^2.0.4", + "phar-io/version": "^3.2.1", + "php": ">=7.3", + "phpunit/php-code-coverage": "^9.2.32", + "phpunit/php-file-iterator": "^3.0.6", + "phpunit/php-invoker": "^3.1.1", + "phpunit/php-text-template": "^2.0.4", + "phpunit/php-timer": "^5.0.3", + "sebastian/cli-parser": "^1.0.2", + "sebastian/code-unit": "^1.0.8", + "sebastian/comparator": "^4.0.8", + "sebastian/diff": "^4.0.6", + "sebastian/environment": "^5.1.5", + "sebastian/exporter": "^4.0.6", + "sebastian/global-state": "^5.0.7", + "sebastian/object-enumerator": "^4.0.4", + "sebastian/resource-operations": "^3.0.4", + "sebastian/type": "^3.2.1", + "sebastian/version": "^3.0.2" }, "suggest": { - "ext-xdebug": "*", - "phpunit/php-invoker": "^1.1" + "ext-soap": "To be able to generate mocks based on WSDL files", + "ext-xdebug": "PHP extension that provides line coverage as well as branch and path coverage" }, "bin": [ "phpunit" @@ -819,10 +923,13 @@ "type": "library", "extra": { "branch-alias": { - "dev-master": "6.5.x-dev" + "dev-master": "9.6-dev" } }, "autoload": { + "files": [ + "src/Framework/Assert/Functions.php" + ], "classmap": [ "src/" ] @@ -845,41 +952,59 @@ "testing", "xunit" ], - "time": "2018-09-08T15:10:43+00:00" + "support": { + "issues": "https://github.com/sebastianbergmann/phpunit/issues", + "security": "https://github.com/sebastianbergmann/phpunit/security/policy", + "source": "https://github.com/sebastianbergmann/phpunit/tree/9.6.23" + }, + "funding": [ + { + "url": "https://phpunit.de/sponsors.html", + "type": "custom" + }, + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + }, + { + "url": "https://liberapay.com/sebastianbergmann", + "type": "liberapay" + }, + { + "url": "https://thanks.dev/u/gh/sebastianbergmann", + "type": "thanks_dev" + }, + { + "url": "https://tidelift.com/funding/github/packagist/phpunit/phpunit", + "type": "tidelift" + } + ], + "time": "2025-05-02T06:40:34+00:00" }, { - "name": "phpunit/phpunit-mock-objects", - "version": "5.0.10", + "name": "sebastian/cli-parser", + "version": "1.0.2", "source": { "type": "git", - "url": "https://github.com/sebastianbergmann/phpunit-mock-objects.git", - "reference": "cd1cf05c553ecfec36b170070573e540b67d3f1f" + "url": "https://github.com/sebastianbergmann/cli-parser.git", + "reference": "2b56bea83a09de3ac06bb18b92f068e60cc6f50b" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/phpunit-mock-objects/zipball/cd1cf05c553ecfec36b170070573e540b67d3f1f", - "reference": "cd1cf05c553ecfec36b170070573e540b67d3f1f", + "url": "https://api.github.com/repos/sebastianbergmann/cli-parser/zipball/2b56bea83a09de3ac06bb18b92f068e60cc6f50b", + "reference": "2b56bea83a09de3ac06bb18b92f068e60cc6f50b", "shasum": "" }, "require": { - "doctrine/instantiator": "^1.0.5", - "php": "^7.0", - "phpunit/php-text-template": "^1.2.1", - "sebastian/exporter": "^3.1" - }, - "conflict": { - "phpunit/phpunit": "<6.0" + "php": ">=7.3" }, "require-dev": { - "phpunit/phpunit": "^6.5.11" - }, - "suggest": { - "ext-soap": "*" + "phpunit/phpunit": "^9.3" }, "type": "library", "extra": { "branch-alias": { - "dev-master": "5.0.x-dev" + "dev-master": "1.0-dev" } }, "autoload": { @@ -898,38 +1023,100 @@ "role": "lead" } ], - "description": "Mock Object library for PHPUnit", - "homepage": "https://github.com/sebastianbergmann/phpunit-mock-objects/", - "keywords": [ - "mock", - "xunit" + "description": "Library for parsing CLI options", + "homepage": "https://github.com/sebastianbergmann/cli-parser", + "support": { + "issues": "https://github.com/sebastianbergmann/cli-parser/issues", + "source": "https://github.com/sebastianbergmann/cli-parser/tree/1.0.2" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } ], - "time": "2018-08-09T05:50:03+00:00" + "time": "2024-03-02T06:27:43+00:00" }, { - "name": "sebastian/code-unit-reverse-lookup", - "version": "1.0.1", + "name": "sebastian/code-unit", + "version": "1.0.8", "source": { "type": "git", - "url": "https://github.com/sebastianbergmann/code-unit-reverse-lookup.git", - "reference": "4419fcdb5eabb9caa61a27c7a1db532a6b55dd18" + "url": "https://github.com/sebastianbergmann/code-unit.git", + "reference": "1fc9f64c0927627ef78ba436c9b17d967e68e120" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/code-unit-reverse-lookup/zipball/4419fcdb5eabb9caa61a27c7a1db532a6b55dd18", - "reference": "4419fcdb5eabb9caa61a27c7a1db532a6b55dd18", + "url": "https://api.github.com/repos/sebastianbergmann/code-unit/zipball/1fc9f64c0927627ef78ba436c9b17d967e68e120", + "reference": "1fc9f64c0927627ef78ba436c9b17d967e68e120", "shasum": "" }, "require": { - "php": "^5.6 || ^7.0" + "php": ">=7.3" }, "require-dev": { - "phpunit/phpunit": "^5.7 || ^6.0" + "phpunit/phpunit": "^9.3" }, "type": "library", "extra": { "branch-alias": { - "dev-master": "1.0.x-dev" + "dev-master": "1.0-dev" + } + }, + "autoload": { + "classmap": [ + "src/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de", + "role": "lead" + } + ], + "description": "Collection of value objects that represent the PHP code units", + "homepage": "https://github.com/sebastianbergmann/code-unit", + "support": { + "issues": "https://github.com/sebastianbergmann/code-unit/issues", + "source": "https://github.com/sebastianbergmann/code-unit/tree/1.0.8" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2020-10-26T13:08:54+00:00" + }, + { + "name": "sebastian/code-unit-reverse-lookup", + "version": "2.0.3", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/code-unit-reverse-lookup.git", + "reference": "ac91f01ccec49fb77bdc6fd1e548bc70f7faa3e5" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/code-unit-reverse-lookup/zipball/ac91f01ccec49fb77bdc6fd1e548bc70f7faa3e5", + "reference": "ac91f01ccec49fb77bdc6fd1e548bc70f7faa3e5", + "shasum": "" + }, + "require": { + "php": ">=7.3" + }, + "require-dev": { + "phpunit/phpunit": "^9.3" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "2.0-dev" } }, "autoload": { @@ -949,34 +1136,44 @@ ], "description": "Looks up which function or method a line of code belongs to", "homepage": "https://github.com/sebastianbergmann/code-unit-reverse-lookup/", - "time": "2017-03-04T06:30:41+00:00" + "support": { + "issues": "https://github.com/sebastianbergmann/code-unit-reverse-lookup/issues", + "source": "https://github.com/sebastianbergmann/code-unit-reverse-lookup/tree/2.0.3" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2020-09-28T05:30:19+00:00" }, { "name": "sebastian/comparator", - "version": "2.1.3", + "version": "4.0.8", "source": { "type": "git", "url": "https://github.com/sebastianbergmann/comparator.git", - "reference": "34369daee48eafb2651bea869b4b15d75ccc35f9" + "reference": "fa0f136dd2334583309d32b62544682ee972b51a" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/comparator/zipball/34369daee48eafb2651bea869b4b15d75ccc35f9", - "reference": "34369daee48eafb2651bea869b4b15d75ccc35f9", + "url": "https://api.github.com/repos/sebastianbergmann/comparator/zipball/fa0f136dd2334583309d32b62544682ee972b51a", + "reference": "fa0f136dd2334583309d32b62544682ee972b51a", "shasum": "" }, "require": { - "php": "^7.0", - "sebastian/diff": "^2.0 || ^3.0", - "sebastian/exporter": "^3.1" + "php": ">=7.3", + "sebastian/diff": "^4.0", + "sebastian/exporter": "^4.0" }, "require-dev": { - "phpunit/phpunit": "^6.4" + "phpunit/phpunit": "^9.3" }, "type": "library", "extra": { "branch-alias": { - "dev-master": "2.1.x-dev" + "dev-master": "4.0-dev" } }, "autoload": { @@ -989,6 +1186,10 @@ "BSD-3-Clause" ], "authors": [ + { + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de" + }, { "name": "Jeff Welch", "email": "whatthejeff@gmail.com" @@ -1000,10 +1201,6 @@ { "name": "Bernhard Schussek", "email": "bschussek@2bepublished.at" - }, - { - "name": "Sebastian Bergmann", - "email": "sebastian@phpunit.de" } ], "description": "Provides the functionality to compare PHP values for equality", @@ -1013,27 +1210,38 @@ "compare", "equality" ], - "time": "2018-02-01T13:46:46+00:00" + "support": { + "issues": "https://github.com/sebastianbergmann/comparator/issues", + "source": "https://github.com/sebastianbergmann/comparator/tree/4.0.8" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2022-09-14T12:41:17+00:00" }, { - "name": "sebastian/diff", - "version": "2.0.1", + "name": "sebastian/complexity", + "version": "2.0.3", "source": { "type": "git", - "url": "https://github.com/sebastianbergmann/diff.git", - "reference": "347c1d8b49c5c3ee30c7040ea6fc446790e6bddd" + "url": "https://github.com/sebastianbergmann/complexity.git", + "reference": "25f207c40d62b8b7aa32f5ab026c53561964053a" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/diff/zipball/347c1d8b49c5c3ee30c7040ea6fc446790e6bddd", - "reference": "347c1d8b49c5c3ee30c7040ea6fc446790e6bddd", + "url": "https://api.github.com/repos/sebastianbergmann/complexity/zipball/25f207c40d62b8b7aa32f5ab026c53561964053a", + "reference": "25f207c40d62b8b7aa32f5ab026c53561964053a", "shasum": "" }, "require": { - "php": "^7.0" + "nikic/php-parser": "^4.18 || ^5.0", + "php": ">=7.3" }, "require-dev": { - "phpunit/phpunit": "^6.2" + "phpunit/phpunit": "^9.3" }, "type": "library", "extra": { @@ -1052,45 +1260,118 @@ ], "authors": [ { - "name": "Kore Nordmann", - "email": "mail@kore-nordmann.de" - }, + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de", + "role": "lead" + } + ], + "description": "Library for calculating the complexity of PHP code units", + "homepage": "https://github.com/sebastianbergmann/complexity", + "support": { + "issues": "https://github.com/sebastianbergmann/complexity/issues", + "source": "https://github.com/sebastianbergmann/complexity/tree/2.0.3" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2023-12-22T06:19:30+00:00" + }, + { + "name": "sebastian/diff", + "version": "4.0.6", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/diff.git", + "reference": "ba01945089c3a293b01ba9badc29ad55b106b0bc" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/diff/zipball/ba01945089c3a293b01ba9badc29ad55b106b0bc", + "reference": "ba01945089c3a293b01ba9badc29ad55b106b0bc", + "shasum": "" + }, + "require": { + "php": ">=7.3" + }, + "require-dev": { + "phpunit/phpunit": "^9.3", + "symfony/process": "^4.2 || ^5" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "4.0-dev" + } + }, + "autoload": { + "classmap": [ + "src/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ { "name": "Sebastian Bergmann", "email": "sebastian@phpunit.de" + }, + { + "name": "Kore Nordmann", + "email": "mail@kore-nordmann.de" } ], "description": "Diff implementation", "homepage": "https://github.com/sebastianbergmann/diff", "keywords": [ - "diff" + "diff", + "udiff", + "unidiff", + "unified diff" ], - "time": "2017-08-03T08:09:46+00:00" + "support": { + "issues": "https://github.com/sebastianbergmann/diff/issues", + "source": "https://github.com/sebastianbergmann/diff/tree/4.0.6" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2024-03-02T06:30:58+00:00" }, { "name": "sebastian/environment", - "version": "3.1.0", + "version": "5.1.5", "source": { "type": "git", "url": "https://github.com/sebastianbergmann/environment.git", - "reference": "cd0871b3975fb7fc44d11314fd1ee20925fce4f5" + "reference": "830c43a844f1f8d5b7a1f6d6076b784454d8b7ed" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/environment/zipball/cd0871b3975fb7fc44d11314fd1ee20925fce4f5", - "reference": "cd0871b3975fb7fc44d11314fd1ee20925fce4f5", + "url": "https://api.github.com/repos/sebastianbergmann/environment/zipball/830c43a844f1f8d5b7a1f6d6076b784454d8b7ed", + "reference": "830c43a844f1f8d5b7a1f6d6076b784454d8b7ed", "shasum": "" }, "require": { - "php": "^7.0" + "php": ">=7.3" }, "require-dev": { - "phpunit/phpunit": "^6.1" + "phpunit/phpunit": "^9.3" + }, + "suggest": { + "ext-posix": "*" }, "type": "library", "extra": { "branch-alias": { - "dev-master": "3.1.x-dev" + "dev-master": "5.1-dev" } }, "autoload": { @@ -1115,34 +1396,44 @@ "environment", "hhvm" ], - "time": "2017-07-01T08:51:00+00:00" + "support": { + "issues": "https://github.com/sebastianbergmann/environment/issues", + "source": "https://github.com/sebastianbergmann/environment/tree/5.1.5" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2023-02-03T06:03:51+00:00" }, { "name": "sebastian/exporter", - "version": "3.1.0", + "version": "4.0.6", "source": { "type": "git", "url": "https://github.com/sebastianbergmann/exporter.git", - "reference": "234199f4528de6d12aaa58b612e98f7d36adb937" + "reference": "78c00df8f170e02473b682df15bfcdacc3d32d72" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/exporter/zipball/234199f4528de6d12aaa58b612e98f7d36adb937", - "reference": "234199f4528de6d12aaa58b612e98f7d36adb937", + "url": "https://api.github.com/repos/sebastianbergmann/exporter/zipball/78c00df8f170e02473b682df15bfcdacc3d32d72", + "reference": "78c00df8f170e02473b682df15bfcdacc3d32d72", "shasum": "" }, "require": { - "php": "^7.0", - "sebastian/recursion-context": "^3.0" + "php": ">=7.3", + "sebastian/recursion-context": "^4.0" }, "require-dev": { "ext-mbstring": "*", - "phpunit/phpunit": "^6.0" + "phpunit/phpunit": "^9.3" }, "type": "library", "extra": { "branch-alias": { - "dev-master": "3.1.x-dev" + "dev-master": "4.0-dev" } }, "autoload": { @@ -1155,6 +1446,10 @@ "BSD-3-Clause" ], "authors": [ + { + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de" + }, { "name": "Jeff Welch", "email": "whatthejeff@gmail.com" @@ -1163,46 +1458,55 @@ "name": "Volker Dusch", "email": "github@wallbash.com" }, - { - "name": "Bernhard Schussek", - "email": "bschussek@2bepublished.at" - }, - { - "name": "Sebastian Bergmann", - "email": "sebastian@phpunit.de" - }, { "name": "Adam Harvey", "email": "aharvey@php.net" + }, + { + "name": "Bernhard Schussek", + "email": "bschussek@gmail.com" } ], "description": "Provides the functionality to export PHP variables for visualization", - "homepage": "http://www.github.com/sebastianbergmann/exporter", + "homepage": "https://www.github.com/sebastianbergmann/exporter", "keywords": [ "export", "exporter" ], - "time": "2017-04-03T13:19:02+00:00" + "support": { + "issues": "https://github.com/sebastianbergmann/exporter/issues", + "source": "https://github.com/sebastianbergmann/exporter/tree/4.0.6" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2024-03-02T06:33:00+00:00" }, { "name": "sebastian/global-state", - "version": "2.0.0", + "version": "5.0.7", "source": { "type": "git", "url": "https://github.com/sebastianbergmann/global-state.git", - "reference": "e8ba02eed7bbbb9e59e43dedd3dddeff4a56b0c4" + "reference": "bca7df1f32ee6fe93b4d4a9abbf69e13a4ada2c9" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/global-state/zipball/e8ba02eed7bbbb9e59e43dedd3dddeff4a56b0c4", - "reference": "e8ba02eed7bbbb9e59e43dedd3dddeff4a56b0c4", + "url": "https://api.github.com/repos/sebastianbergmann/global-state/zipball/bca7df1f32ee6fe93b4d4a9abbf69e13a4ada2c9", + "reference": "bca7df1f32ee6fe93b4d4a9abbf69e13a4ada2c9", "shasum": "" }, "require": { - "php": "^7.0" + "php": ">=7.3", + "sebastian/object-reflector": "^2.0", + "sebastian/recursion-context": "^4.0" }, "require-dev": { - "phpunit/phpunit": "^6.0" + "ext-dom": "*", + "phpunit/phpunit": "^9.3" }, "suggest": { "ext-uopz": "*" @@ -1210,7 +1514,7 @@ "type": "library", "extra": { "branch-alias": { - "dev-master": "2.0-dev" + "dev-master": "5.0-dev" } }, "autoload": { @@ -1233,34 +1537,101 @@ "keywords": [ "global state" ], - "time": "2017-04-27T15:39:26+00:00" + "support": { + "issues": "https://github.com/sebastianbergmann/global-state/issues", + "source": "https://github.com/sebastianbergmann/global-state/tree/5.0.7" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2024-03-02T06:35:11+00:00" }, { - "name": "sebastian/object-enumerator", - "version": "3.0.3", + "name": "sebastian/lines-of-code", + "version": "1.0.4", "source": { "type": "git", - "url": "https://github.com/sebastianbergmann/object-enumerator.git", - "reference": "7cfd9e65d11ffb5af41198476395774d4c8a84c5" + "url": "https://github.com/sebastianbergmann/lines-of-code.git", + "reference": "e1e4a170560925c26d424b6a03aed157e7dcc5c5" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/object-enumerator/zipball/7cfd9e65d11ffb5af41198476395774d4c8a84c5", - "reference": "7cfd9e65d11ffb5af41198476395774d4c8a84c5", + "url": "https://api.github.com/repos/sebastianbergmann/lines-of-code/zipball/e1e4a170560925c26d424b6a03aed157e7dcc5c5", + "reference": "e1e4a170560925c26d424b6a03aed157e7dcc5c5", "shasum": "" }, "require": { - "php": "^7.0", - "sebastian/object-reflector": "^1.1.1", - "sebastian/recursion-context": "^3.0" + "nikic/php-parser": "^4.18 || ^5.0", + "php": ">=7.3" }, "require-dev": { - "phpunit/phpunit": "^6.0" + "phpunit/phpunit": "^9.3" }, "type": "library", "extra": { "branch-alias": { - "dev-master": "3.0.x-dev" + "dev-master": "1.0-dev" + } + }, + "autoload": { + "classmap": [ + "src/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de", + "role": "lead" + } + ], + "description": "Library for counting the lines of code in PHP source code", + "homepage": "https://github.com/sebastianbergmann/lines-of-code", + "support": { + "issues": "https://github.com/sebastianbergmann/lines-of-code/issues", + "source": "https://github.com/sebastianbergmann/lines-of-code/tree/1.0.4" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2023-12-22T06:20:34+00:00" + }, + { + "name": "sebastian/object-enumerator", + "version": "4.0.4", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/object-enumerator.git", + "reference": "5c9eeac41b290a3712d88851518825ad78f45c71" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/object-enumerator/zipball/5c9eeac41b290a3712d88851518825ad78f45c71", + "reference": "5c9eeac41b290a3712d88851518825ad78f45c71", + "shasum": "" + }, + "require": { + "php": ">=7.3", + "sebastian/object-reflector": "^2.0", + "sebastian/recursion-context": "^4.0" + }, + "require-dev": { + "phpunit/phpunit": "^9.3" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "4.0-dev" } }, "autoload": { @@ -1280,32 +1651,42 @@ ], "description": "Traverses array structures and object graphs to enumerate all referenced objects", "homepage": "https://github.com/sebastianbergmann/object-enumerator/", - "time": "2017-08-03T12:35:26+00:00" + "support": { + "issues": "https://github.com/sebastianbergmann/object-enumerator/issues", + "source": "https://github.com/sebastianbergmann/object-enumerator/tree/4.0.4" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2020-10-26T13:12:34+00:00" }, { "name": "sebastian/object-reflector", - "version": "1.1.1", + "version": "2.0.4", "source": { "type": "git", "url": "https://github.com/sebastianbergmann/object-reflector.git", - "reference": "773f97c67f28de00d397be301821b06708fca0be" + "reference": "b4f479ebdbf63ac605d183ece17d8d7fe49c15c7" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/object-reflector/zipball/773f97c67f28de00d397be301821b06708fca0be", - "reference": "773f97c67f28de00d397be301821b06708fca0be", + "url": "https://api.github.com/repos/sebastianbergmann/object-reflector/zipball/b4f479ebdbf63ac605d183ece17d8d7fe49c15c7", + "reference": "b4f479ebdbf63ac605d183ece17d8d7fe49c15c7", "shasum": "" }, "require": { - "php": "^7.0" + "php": ">=7.3" }, "require-dev": { - "phpunit/phpunit": "^6.0" + "phpunit/phpunit": "^9.3" }, "type": "library", "extra": { "branch-alias": { - "dev-master": "1.1-dev" + "dev-master": "2.0-dev" } }, "autoload": { @@ -1325,32 +1706,42 @@ ], "description": "Allows reflection of object attributes, including inherited and non-public ones", "homepage": "https://github.com/sebastianbergmann/object-reflector/", - "time": "2017-03-29T09:07:27+00:00" + "support": { + "issues": "https://github.com/sebastianbergmann/object-reflector/issues", + "source": "https://github.com/sebastianbergmann/object-reflector/tree/2.0.4" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2020-10-26T13:14:26+00:00" }, { "name": "sebastian/recursion-context", - "version": "3.0.0", + "version": "4.0.5", "source": { "type": "git", "url": "https://github.com/sebastianbergmann/recursion-context.git", - "reference": "5b0cd723502bac3b006cbf3dbf7a1e3fcefe4fa8" + "reference": "e75bd0f07204fec2a0af9b0f3cfe97d05f92efc1" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/recursion-context/zipball/5b0cd723502bac3b006cbf3dbf7a1e3fcefe4fa8", - "reference": "5b0cd723502bac3b006cbf3dbf7a1e3fcefe4fa8", + "url": "https://api.github.com/repos/sebastianbergmann/recursion-context/zipball/e75bd0f07204fec2a0af9b0f3cfe97d05f92efc1", + "reference": "e75bd0f07204fec2a0af9b0f3cfe97d05f92efc1", "shasum": "" }, "require": { - "php": "^7.0" + "php": ">=7.3" }, "require-dev": { - "phpunit/phpunit": "^6.0" + "phpunit/phpunit": "^9.3" }, "type": "library", "extra": { "branch-alias": { - "dev-master": "3.0.x-dev" + "dev-master": "4.0-dev" } }, "autoload": { @@ -1363,44 +1754,57 @@ "BSD-3-Clause" ], "authors": [ - { - "name": "Jeff Welch", - "email": "whatthejeff@gmail.com" - }, { "name": "Sebastian Bergmann", "email": "sebastian@phpunit.de" }, + { + "name": "Jeff Welch", + "email": "whatthejeff@gmail.com" + }, { "name": "Adam Harvey", "email": "aharvey@php.net" } ], "description": "Provides functionality to recursively process PHP variables", - "homepage": "http://www.github.com/sebastianbergmann/recursion-context", - "time": "2017-03-03T06:23:57+00:00" + "homepage": "https://github.com/sebastianbergmann/recursion-context", + "support": { + "issues": "https://github.com/sebastianbergmann/recursion-context/issues", + "source": "https://github.com/sebastianbergmann/recursion-context/tree/4.0.5" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2023-02-03T06:07:39+00:00" }, { "name": "sebastian/resource-operations", - "version": "1.0.0", + "version": "3.0.4", "source": { "type": "git", "url": "https://github.com/sebastianbergmann/resource-operations.git", - "reference": "ce990bb21759f94aeafd30209e8cfcdfa8bc3f52" + "reference": "05d5692a7993ecccd56a03e40cd7e5b09b1d404e" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/resource-operations/zipball/ce990bb21759f94aeafd30209e8cfcdfa8bc3f52", - "reference": "ce990bb21759f94aeafd30209e8cfcdfa8bc3f52", + "url": "https://api.github.com/repos/sebastianbergmann/resource-operations/zipball/05d5692a7993ecccd56a03e40cd7e5b09b1d404e", + "reference": "05d5692a7993ecccd56a03e40cd7e5b09b1d404e", "shasum": "" }, "require": { - "php": ">=5.6.0" + "php": ">=7.3" + }, + "require-dev": { + "phpunit/phpunit": "^9.0" }, "type": "library", "extra": { "branch-alias": { - "dev-master": "1.0.x-dev" + "dev-main": "3.0-dev" } }, "autoload": { @@ -1420,29 +1824,94 @@ ], "description": "Provides a list of PHP built-in functions that operate on resources", "homepage": "https://www.github.com/sebastianbergmann/resource-operations", - "time": "2015-07-28T20:34:47+00:00" + "support": { + "source": "https://github.com/sebastianbergmann/resource-operations/tree/3.0.4" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2024-03-14T16:00:52+00:00" }, { - "name": "sebastian/version", - "version": "2.0.1", + "name": "sebastian/type", + "version": "3.2.1", "source": { "type": "git", - "url": "https://github.com/sebastianbergmann/version.git", - "reference": "99732be0ddb3361e16ad77b68ba41efc8e979019" + "url": "https://github.com/sebastianbergmann/type.git", + "reference": "75e2c2a32f5e0b3aef905b9ed0b179b953b3d7c7" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/version/zipball/99732be0ddb3361e16ad77b68ba41efc8e979019", - "reference": "99732be0ddb3361e16ad77b68ba41efc8e979019", + "url": "https://api.github.com/repos/sebastianbergmann/type/zipball/75e2c2a32f5e0b3aef905b9ed0b179b953b3d7c7", + "reference": "75e2c2a32f5e0b3aef905b9ed0b179b953b3d7c7", "shasum": "" }, "require": { - "php": ">=5.6" + "php": ">=7.3" + }, + "require-dev": { + "phpunit/phpunit": "^9.5" }, "type": "library", "extra": { "branch-alias": { - "dev-master": "2.0.x-dev" + "dev-master": "3.2-dev" + } + }, + "autoload": { + "classmap": [ + "src/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de", + "role": "lead" + } + ], + "description": "Collection of value objects that represent the types of the PHP type system", + "homepage": "https://github.com/sebastianbergmann/type", + "support": { + "issues": "https://github.com/sebastianbergmann/type/issues", + "source": "https://github.com/sebastianbergmann/type/tree/3.2.1" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2023-02-03T06:13:03+00:00" + }, + { + "name": "sebastian/version", + "version": "3.0.2", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/version.git", + "reference": "c6c1022351a901512170118436c764e473f6de8c" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/version/zipball/c6c1022351a901512170118436c764e473f6de8c", + "reference": "c6c1022351a901512170118436c764e473f6de8c", + "shasum": "" + }, + "require": { + "php": ">=7.3" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "3.0-dev" } }, "autoload": { @@ -1463,27 +1932,121 @@ ], "description": "Library that helps with managing the version number of Git-hosted PHP projects", "homepage": "https://github.com/sebastianbergmann/version", - "time": "2016-10-03T07:35:21+00:00" + "support": { + "issues": "https://github.com/sebastianbergmann/version/issues", + "source": "https://github.com/sebastianbergmann/version/tree/3.0.2" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2020-09-28T06:39:44+00:00" }, { - "name": "theseer/tokenizer", - "version": "1.1.0", + "name": "squizlabs/php_codesniffer", + "version": "3.13.0", "source": { "type": "git", - "url": "https://github.com/theseer/tokenizer.git", - "reference": "cb2f008f3f05af2893a87208fe6a6c4985483f8b" + "url": "https://github.com/PHPCSStandards/PHP_CodeSniffer.git", + "reference": "65ff2489553b83b4597e89c3b8b721487011d186" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/theseer/tokenizer/zipball/cb2f008f3f05af2893a87208fe6a6c4985483f8b", - "reference": "cb2f008f3f05af2893a87208fe6a6c4985483f8b", + "url": "https://api.github.com/repos/PHPCSStandards/PHP_CodeSniffer/zipball/65ff2489553b83b4597e89c3b8b721487011d186", + "reference": "65ff2489553b83b4597e89c3b8b721487011d186", + "shasum": "" + }, + "require": { + "ext-simplexml": "*", + "ext-tokenizer": "*", + "ext-xmlwriter": "*", + "php": ">=5.4.0" + }, + "require-dev": { + "phpunit/phpunit": "^4.0 || ^5.0 || ^6.0 || ^7.0 || ^8.0 || ^9.3.4" + }, + "bin": [ + "bin/phpcbf", + "bin/phpcs" + ], + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "3.x-dev" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Greg Sherwood", + "role": "Former lead" + }, + { + "name": "Juliette Reinders Folmer", + "role": "Current lead" + }, + { + "name": "Contributors", + "homepage": "https://github.com/PHPCSStandards/PHP_CodeSniffer/graphs/contributors" + } + ], + "description": "PHP_CodeSniffer tokenizes PHP, JavaScript and CSS files and detects violations of a defined set of coding standards.", + "homepage": "https://github.com/PHPCSStandards/PHP_CodeSniffer", + "keywords": [ + "phpcs", + "standards", + "static analysis" + ], + "support": { + "issues": "https://github.com/PHPCSStandards/PHP_CodeSniffer/issues", + "security": "https://github.com/PHPCSStandards/PHP_CodeSniffer/security/policy", + "source": "https://github.com/PHPCSStandards/PHP_CodeSniffer", + "wiki": "https://github.com/PHPCSStandards/PHP_CodeSniffer/wiki" + }, + "funding": [ + { + "url": "https://github.com/PHPCSStandards", + "type": "github" + }, + { + "url": "https://github.com/jrfnl", + "type": "github" + }, + { + "url": "https://opencollective.com/php_codesniffer", + "type": "open_collective" + }, + { + "url": "https://thanks.dev/u/gh/phpcsstandards", + "type": "thanks_dev" + } + ], + "time": "2025-05-11T03:36:00+00:00" + }, + { + "name": "theseer/tokenizer", + "version": "1.2.3", + "source": { + "type": "git", + "url": "https://github.com/theseer/tokenizer.git", + "reference": "737eda637ed5e28c3413cb1ebe8bb52cbf1ca7a2" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/theseer/tokenizer/zipball/737eda637ed5e28c3413cb1ebe8bb52cbf1ca7a2", + "reference": "737eda637ed5e28c3413cb1ebe8bb52cbf1ca7a2", "shasum": "" }, "require": { "ext-dom": "*", "ext-tokenizer": "*", "ext-xmlwriter": "*", - "php": "^7.0" + "php": "^7.2 || ^8.0" }, "type": "library", "autoload": { @@ -1503,68 +2066,27 @@ } ], "description": "A small library for converting tokenized PHP source code into XML and potentially other formats", - "time": "2017-04-07T12:08:54+00:00" - }, - { - "name": "webmozart/assert", - "version": "1.3.0", - "source": { - "type": "git", - "url": "https://github.com/webmozart/assert.git", - "reference": "0df1908962e7a3071564e857d86874dad1ef204a" + "support": { + "issues": "https://github.com/theseer/tokenizer/issues", + "source": "https://github.com/theseer/tokenizer/tree/1.2.3" }, - "dist": { - "type": "zip", - "url": "https://api.github.com/repos/webmozart/assert/zipball/0df1908962e7a3071564e857d86874dad1ef204a", - "reference": "0df1908962e7a3071564e857d86874dad1ef204a", - "shasum": "" - }, - "require": { - "php": "^5.3.3 || ^7.0" - }, - "require-dev": { - "phpunit/phpunit": "^4.6", - "sebastian/version": "^1.0.1" - }, - "type": "library", - "extra": { - "branch-alias": { - "dev-master": "1.3-dev" - } - }, - "autoload": { - "psr-4": { - "Webmozart\\Assert\\": "src/" - } - }, - "notification-url": "https://packagist.org/downloads/", - "license": [ - "MIT" - ], - "authors": [ + "funding": [ { - "name": "Bernhard Schussek", - "email": "bschussek@gmail.com" + "url": "https://github.com/theseer", + "type": "github" } ], - "description": "Assertions to validate method input/output with nice error messages.", - "keywords": [ - "assert", - "check", - "validate" - ], - "time": "2018-01-29T19:49:41+00:00" + "time": "2024-03-03T12:36:25+00:00" } ], "aliases": [], "minimum-stability": "stable", - "stability-flags": [], + "stability-flags": {}, "prefer-stable": false, "prefer-lowest": false, "platform": { - "php": "^7.0", - "ext-pcntl": "*", - "ext-redis": "*" + "php": ">=8.1" }, - "platform-dev": [] + "platform-dev": {}, + "plugin-api-version": "2.6.0" } diff --git a/demo/bad_job.php b/demo/bad_job.php deleted file mode 100644 index cd719cc..0000000 --- a/demo/bad_job.php +++ /dev/null @@ -1,8 +0,0 @@ -isTracking()) { die("Resque is not tracking the status of this job.\n"); } @@ -20,4 +23,4 @@ echo "Tracking status of " . $argv[1] . ". Press [break] to stop.\n\n"; while (true) { fwrite(STDOUT, "Status of " . $argv[1] . " is: " . $status->get() . "\n"); sleep(1); -} \ No newline at end of file +} diff --git a/demo/init.php b/examples/Init.php similarity index 95% rename from demo/init.php rename to examples/Init.php index bdad7e5..11d4833 100644 --- a/demo/init.php +++ b/examples/Init.php @@ -1,4 +1,7 @@ /var/log/resque/worker_[QUEUE].log &'" as uid [UID] and gid [GID] - stop program = "/bin/sh -c 'kill -s QUIT `cat /var/run/resque/worker_[QUEUE].pid` && rm -f /var/run/resque/worker_[QUEUE].pid; exit 0;'" - if totalmem is greater than 300 MB for 10 cycles then restart # eating up memory? - group resque_workers \ No newline at end of file diff --git a/lib/Resque/Job/Factory.php b/lib/Resque/Job/Factory.php deleted file mode 100644 index cf17294..0000000 --- a/lib/Resque/Job/Factory.php +++ /dev/null @@ -1,32 +0,0 @@ -args = $args; - $instance->queue = $queue; - return $instance; - } -} diff --git a/lib/Resque/Job/FactoryInterface.php b/lib/Resque/Job/FactoryInterface.php deleted file mode 100644 index a1203e1..0000000 --- a/lib/Resque/Job/FactoryInterface.php +++ /dev/null @@ -1,12 +0,0 @@ - - * @license http://www.opensource.org/licenses/mit-license.php - */ - -class Resque_RedisException extends Resque_Exception -{ - -} diff --git a/phpunit.xml.dist b/phpunit.xml similarity index 58% rename from phpunit.xml.dist rename to phpunit.xml index 61d2d7b..c718913 100644 --- a/phpunit.xml.dist +++ b/phpunit.xml @@ -1,24 +1,17 @@ - ./test/Resque/ + ./tests/Resque/ - - - - ./lib/Resque/ - - \ No newline at end of file diff --git a/ruleset.xml b/ruleset.xml new file mode 100644 index 0000000..6bd9f60 --- /dev/null +++ b/ruleset.xml @@ -0,0 +1,29 @@ + + + PHP8.2 Ruleset + + . + vendor + tests/ + + + + + + + + + + + + + + + + + + + + + diff --git a/lib/Resque/Event.php b/src/Resque/Event.php similarity index 88% rename from lib/Resque/Event.php rename to src/Resque/Event.php index 2c8f182..f445375 100644 --- a/lib/Resque/Event.php +++ b/src/Resque/Event.php @@ -1,14 +1,16 @@ + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -class Resque_Event +class Event { /** * @var array Array containing all registered callbacks, indexked by event name. @@ -20,6 +22,7 @@ class Resque_Event * * @param string $event Name of event to be raised. * @param mixed $data Optional, any data that should be passed to each callback. + * * @return true */ public static function trigger($event, $data = null) @@ -36,7 +39,8 @@ class Resque_Event if (!is_callable($callback)) { continue; } - call_user_func_array($callback, $data); + + call_user_func_array($callback, array_values($data)); } return true; @@ -46,7 +50,8 @@ class Resque_Event * Listen in on a given event to have a specified callback fired. * * @param string $event Name of event to listen on. - * @param mixed $callback Any callback callable by call_user_func_array. + * @param mixed $callback Any callback callable by call_user_func_array + * * @return true */ public static function listen($event, $callback) @@ -64,6 +69,7 @@ class Resque_Event * * @param string $event Name of event. * @param mixed $callback The callback as defined when listen() was called. + * * @return true */ public static function stopListening($event, $callback) @@ -82,8 +88,10 @@ class Resque_Event /** * Call all registered listeners. + * + * @return void */ - public static function clearListeners() + public static function clearListeners(): void { self::$events = []; } diff --git a/lib/Resque/Exception.php b/src/Resque/Exception.php similarity index 58% rename from lib/Resque/Exception.php rename to src/Resque/Exception.php index fe510ca..7e8b7ae 100644 --- a/lib/Resque/Exception.php +++ b/src/Resque/Exception.php @@ -1,13 +1,15 @@ + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -class Resque_Exception extends Exception +class Exception extends \Exception { } diff --git a/lib/Resque/Failure.php b/src/Resque/Failure/Failure.php similarity index 71% rename from lib/Resque/Failure.php rename to src/Resque/Failure/Failure.php index d73fb6d..1a0b17f 100644 --- a/lib/Resque/Failure.php +++ b/src/Resque/Failure/Failure.php @@ -1,14 +1,16 @@ + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -class Resque_Failure +class Failure { /** * @var string Class name representing the backend to pass failed jobs off to. @@ -20,11 +22,15 @@ class Resque_Failure * * @param array $payload The contents of the job that has just failed. * @param \Exception $exception The exception generated when the job failed to run. - * @param \Resque_Worker $worker Instance of Resque_Worker that was running this job when it failed. + * @param \Resque\Worker $worker Instance of Resque_Worker that was running this job when it failed. * @param string $queue The name of the queue that this job was fetched from. */ - public static function create($payload, Exception $exception, Resque_Worker $worker, $queue) - { + public static function create( + $payload, + \Exception $exception, + \Resque\Worker $worker, + $queue + ) { $backend = self::getBackend(); new $backend($payload, $exception, $worker, $queue); } @@ -32,12 +38,12 @@ class Resque_Failure /** * Return an instance of the backend for saving job failures. * - * @return object|string + * @return string */ public static function getBackend() { - if (self::$backend === null) { - self::$backend = 'Resque_Failure_Redis'; + if (is_null(self::$backend)) { + self::$backend = '\\Resque\\Failure\\ResqueFailureRedis'; } return self::$backend; @@ -49,9 +55,11 @@ class Resque_Failure * It is your responsibility to have the backend class loaded (or autoloaded) * * @param string $backend The class name of the backend to pipe failures to. + * + * @return void */ - public static function setBackend($backend) + public static function setBackend(string $backend): void { self::$backend = $backend; } -} \ No newline at end of file +} diff --git a/lib/Resque/Failure/Interface.php b/src/Resque/Failure/ResqueFailureInterface.php similarity index 81% rename from lib/Resque/Failure/Interface.php rename to src/Resque/Failure/ResqueFailureInterface.php index d6da0e2..980e656 100644 --- a/lib/Resque/Failure/Interface.php +++ b/src/Resque/Failure/ResqueFailureInterface.php @@ -1,13 +1,15 @@ + * @package Resque\Failure + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -interface Resque_Failure_Interface +interface ResqueFailureInterface { /** * Initialize a failed job class and save it (where appropriate). diff --git a/lib/Resque/Failure/Redis.php b/src/Resque/Failure/ResqueFailureRedis.php similarity index 73% rename from lib/Resque/Failure/Redis.php rename to src/Resque/Failure/ResqueFailureRedis.php index dacde7e..33a9cc1 100644 --- a/lib/Resque/Failure/Redis.php +++ b/src/Resque/Failure/ResqueFailureRedis.php @@ -1,13 +1,16 @@ + * @package Resque\Failure + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -class Resque_Failure_Redis implements Resque_Failure_Interface +class ResqueFailureRedis implements ResqueFailureInterface { /** * Initialize a failed job class and save it (where appropriate). @@ -16,12 +19,12 @@ class Resque_Failure_Redis implements Resque_Failure_Interface * @param object $exception Instance of the exception that was thrown by the failed job. * @param object $worker Instance of Resque_Worker that received the job. * @param string $queue The name of the queue the job was fetched from. - * @throws Resque_RedisException + * @throws \Resque\RedisException */ public function __construct($payload, $exception, $worker, $queue) { - $data = new stdClass; - $data->failed_at = strftime('%a %b %d %H:%M:%S %Z %Y'); + $data = new \stdClass(); + $data->failed_at = date('D M d H:i:s T Y'); $data->payload = $payload; $data->exception = get_class($exception); $data->error = $exception->getMessage(); @@ -29,6 +32,6 @@ class Resque_Failure_Redis implements Resque_Failure_Interface $data->worker = (string)$worker; $data->queue = $queue; $data = json_encode($data); - Resque::redis()->rpush('failed', $data); + \Resque\Resque::redis()->rpush('failed', $data); } } diff --git a/lib/Resque/Job/DirtyExitException.php b/src/Resque/Job/DirtyExitException.php similarity index 61% rename from lib/Resque/Job/DirtyExitException.php rename to src/Resque/Job/DirtyExitException.php index 7b1f88b..ba002bc 100644 --- a/lib/Resque/Job/DirtyExitException.php +++ b/src/Resque/Job/DirtyExitException.php @@ -1,13 +1,14 @@ + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -class Resque_Job_DirtyExitException extends RuntimeException +class DirtyExitException extends \RuntimeException { - } diff --git a/lib/Resque/Job/DontCreate.php b/src/Resque/Job/DontCreate.php similarity index 65% rename from lib/Resque/Job/DontCreate.php rename to src/Resque/Job/DontCreate.php index 931ae91..21f4f4d 100644 --- a/lib/Resque/Job/DontCreate.php +++ b/src/Resque/Job/DontCreate.php @@ -1,13 +1,14 @@ + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -class Resque_Job_DontCreate extends Exception +class DontCreate extends \Exception { - } diff --git a/lib/Resque/Job/DontPerform.php b/src/Resque/Job/DontPerform.php similarity index 63% rename from lib/Resque/Job/DontPerform.php rename to src/Resque/Job/DontPerform.php index 69075f4..cc1c029 100644 --- a/lib/Resque/Job/DontPerform.php +++ b/src/Resque/Job/DontPerform.php @@ -1,14 +1,14 @@ + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ - -class Resque_Job_DontPerform extends Exception +class DontPerform extends \Exception { - } diff --git a/src/Resque/Job/Factory.php b/src/Resque/Job/Factory.php new file mode 100644 index 0000000..9ee0b74 --- /dev/null +++ b/src/Resque/Job/Factory.php @@ -0,0 +1,50 @@ + + * @license http://www.opensource.org/licenses/mit-license.php + */ + +class Factory implements FactoryInterface +{ + public ?Job $job; + public string $queue; + public array $args; + + /** + * Create job factory + * + * @param $className + * @param array $args + * @param $queue + * + * @return \Resque\Job\JobInterface + * + * @throws \Resque\Exception + */ + public function create($className, $args, $queue) + { + if (!class_exists($className)) { + throw new \Resque\Exception( + 'Could not find job class ' . $className . '.' + ); + } + + if (!method_exists($className, 'perform')) { + throw new \Resque\Exception( + 'Job class ' . $className . ' does not contain a perform() method.' + ); + } + + $instance = new $className(); + $instance->args = $args; + $instance->queue = $queue; + + return $instance; + } +} diff --git a/src/Resque/Job/FactoryInterface.php b/src/Resque/Job/FactoryInterface.php new file mode 100644 index 0000000..a44d6ac --- /dev/null +++ b/src/Resque/Job/FactoryInterface.php @@ -0,0 +1,22 @@ + + * @license http://www.opensource.org/licenses/mit-license.php + */ +interface FactoryInterface +{ + /** + * @param $className + * @param array $args + * @param $queue + * + * @return \Resque\Job\JobInterface + */ + public function create($className, $args, $queue); +} diff --git a/lib/Resque/Job.php b/src/Resque/Job/Job.php similarity index 68% rename from lib/Resque/Job.php rename to src/Resque/Job/Job.php index d96fadc..8249559 100755 --- a/lib/Resque/Job.php +++ b/src/Resque/Job/Job.php @@ -1,14 +1,16 @@ + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -class Resque_Job +class Job { /** * @var string The name of the queue that this job belongs to. @@ -16,7 +18,7 @@ class Resque_Job public $queue; /** - * @var Resque_Worker Instance of the Resque worker running this job. + * @var \Resque\Worker Instance of the Resque worker running this job. */ public $worker; @@ -26,12 +28,12 @@ class Resque_Job public $payload; /** - * @var object|Resque_JobInterface Instance of the class performing work for this job. + * @var object|\Resque\Job\JobInterface Instance of the class performing work for this job. */ private $instance; /** - * @var Resque_Job_FactoryInterface + * @var \Resque\Job\FactoryInterface */ private $jobFactory; @@ -57,20 +59,21 @@ class Resque_Job * @param string $id Unique identifier for tracking the job. Generated if not supplied. * * @return string + * * @throws \InvalidArgumentException */ - public static function create($queue, $class, $args = null, $monitor = false, $id = null) + public static function create($queue, $class, $args = null, $monitor = false, $id = null): string { if (is_null($id)) { - $id = Resque::generateJobId(); + $id = \Resque\Resque::generateJobId(); } if ($args !== null && !is_array($args)) { - throw new InvalidArgumentException( + throw new \InvalidArgumentException( 'Supplied $args must be an array.' ); } - Resque::push($queue, [ + \Resque\Resque::push($queue, [ 'class' => $class, 'args' => [$args], 'id' => $id, @@ -78,7 +81,7 @@ class Resque_Job ]); if ($monitor) { - Resque_Job_Status::create($id); + Status::create($id); } return $id; @@ -86,43 +89,47 @@ class Resque_Job /** * Find the next available job from the specified queue and return an - * instance of Resque_Job for it. + * instance of \Resque\Job\Job for it. * * @param string $queue The name of the queue to check for a job in. - * @return false|object Null when there aren't any waiting jobs, instance of Resque_Job when a job was found. + * + * @return Job|null Null when there aren't any waiting jobs, instance of \Resque\Job\Job when a job was found. */ - public static function reserve($queue) + public static function reserve($queue): ?Job { - $payload = Resque::pop($queue); + $payload = \Resque\Resque::pop($queue); if (!is_array($payload)) { - return false; + return null; } - return new Resque_Job($queue, $payload); + return new Job($queue, $payload); } /** * Find the next available job from the specified queues using blocking list pop - * and return an instance of Resque_Job for it. + * and return an instance of \Resque\Job\Job for it. * * @param array $queues * @param int $timeout - * @return false|object Null when there aren't any waiting jobs, instance of Resque_Job when a job was found. + * + * @return Job|null Null when there aren't any waiting jobs, instance of \Resque\Job\Job when a job was found. */ - public static function reserveBlocking(array $queues, $timeout = null) + public static function reserveBlocking(array $queues, $timeout = null): ?Job { - $item = Resque::blpop($queues, $timeout); + $item = \Resque\Resque::blpop($queues, $timeout); if (!is_array($item)) { - return false; + return null; } - return new Resque_Job($item['queue'], $item['payload']); + return new Job($item['queue'], $item['payload']); } /** * Update the status of the current job. * - * @param int $status Status constant from Resque_Job_Status indicating the current status of a job. + * @param int $status Status constant from \Resque\Job\Status indicating the current status of a job. + * + * @return bool */ public function updateStatus($status): bool { @@ -130,7 +137,7 @@ class Resque_Job return false; } - $statusInstance = new Resque_Job_Status($this->payload['id']); + $statusInstance = new Status($this->payload['id']); $statusInstance->update($status); return true; } @@ -138,11 +145,11 @@ class Resque_Job /** * Return the status of the current job. * - * @return int The status of the job as one of the Resque_Job_Status constants. + * @return int The status of the job as one of the \Resque\Job\Status constants. */ public function getStatus() { - $status = new Resque_Job_Status($this->payload['id']); + $status = new Status($this->payload['id']); return $status->get(); } @@ -162,11 +169,12 @@ class Resque_Job /** * Get the instantiated object for this job that will be performing work. - * @return Resque_JobInterface Instance of the object that this job belongs to. + * + * @return \Resque\Job\JobInterface Instance of the object that this job belongs to. */ public function getInstance() { - if (!is_null($this->instance)) { + if (isset($this->instance) && !is_null($this->instance)) { return $this->instance; } @@ -180,12 +188,13 @@ class Resque_Job * associated with the job with the supplied arguments. * * @return bool - * @throws Resque_Exception When the job's class could not be found or it does not contain a perform method. + * + * @throws \Resque\Exception When the job's class could not be found or it does not contain a perform method. */ public function perform() { try { - Resque_Event::trigger('beforePerform', $this); + \Resque\Event::trigger('beforePerform', $this); $instance = $this->getInstance(); if (method_exists($instance, 'setUp')) { @@ -198,9 +207,9 @@ class Resque_Job $instance->tearDown(); } - Resque_Event::trigger('afterPerform', $this); - } // beforePerform/setUp have said don't perform this job. Return. - /** @noinspection PhpRedundantCatchClauseInspection */ catch (Resque_Job_DontPerform $e) { + \Resque\Event::trigger('afterPerform', $this); + } catch (DontPerform $e) { + /** @noinspection PhpRedundantCatchClauseInspection */ return false; } @@ -214,29 +223,30 @@ class Resque_Job */ public function fail($exception) { - Resque_Event::trigger('onFailure', [ + \Resque\Event::trigger('onFailure', [ 'exception' => $exception, 'job' => $this, ]); - $this->updateStatus(Resque_Job_Status::STATUS_FAILED); - Resque_Failure::create( + $this->updateStatus(Status::STATUS_FAILED); + \Resque\Failure\Failure::create( $this->payload, $exception, $this->worker, $this->queue ); - Resque_Stat::incr('failed'); - Resque_Stat::incr('failed:' . $this->worker); + \Resque\Stat::incr('failed'); + \Resque\Stat::incr('failed:' . $this->worker); } /** * Re-queue the current job. + * * @return string */ public function recreate() { - $status = new Resque_Job_Status($this->payload['id']); + $status = new Status($this->payload['id']); $monitor = false; if ($status->isTracking()) { $monitor = true; @@ -266,10 +276,11 @@ class Resque_Job } /** - * @param Resque_Job_FactoryInterface $jobFactory - * @return Resque_Job + * @param FactoryInterface $jobFactory + * + * @return Job */ - public function setJobFactory(Resque_Job_FactoryInterface $jobFactory) + public function setJobFactory(FactoryInterface $jobFactory) { $this->jobFactory = $jobFactory; @@ -277,13 +288,14 @@ class Resque_Job } /** - * @return Resque_Job_FactoryInterface + * @return FactoryInterface */ public function getJobFactory() { - if ($this->jobFactory === null) { - $this->jobFactory = new Resque_Job_Factory(); + if (is_null($this->jobFactory)) { + $this->jobFactory = new Factory(); } + return $this->jobFactory; } } diff --git a/lib/Resque/JobInterface.php b/src/Resque/Job/JobInterface.php similarity index 62% rename from lib/Resque/JobInterface.php rename to src/Resque/Job/JobInterface.php index f31281d..ed8b53c 100644 --- a/lib/Resque/JobInterface.php +++ b/src/Resque/Job/JobInterface.php @@ -1,6 +1,8 @@ + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -class Resque_Job_Status +class Status { - const STATUS_WAITING = 1; - const STATUS_RUNNING = 2; - const STATUS_FAILED = 3; - const STATUS_COMPLETE = 4; + public const STATUS_WAITING = 1; + public const STATUS_RUNNING = 2; + public const STATUS_FAILED = 3; + public const STATUS_COMPLETE = 4; /** * @var string The ID of the job this status class refers back to. @@ -57,7 +59,11 @@ class Resque_Job_Status 'updated' => time(), 'started' => time(), ]; - Resque::redis()->set('job:' . $id . ':status', json_encode($statusPacket)); + \Resque\Resque::redis()->set( + 'job:' . $id . ':status', + json_encode($statusPacket), + ['ex' => \Resque\Redis::DEFAULT_REDIS_TTL], + ); } /** @@ -72,7 +78,7 @@ class Resque_Job_Status return false; } - if (!Resque::redis()->exists((string)$this)) { + if (!\Resque\Resque::redis()->exists((string)$this)) { $this->isTracking = false; return false; } @@ -84,7 +90,7 @@ class Resque_Job_Status /** * Update the status indicator for the current job with a new status. * - * @param int The status of the job (see constants in Resque_Job_Status) + * @param int The status of the job (see constants in \Resque\Job\Status) */ public function update($status) { @@ -96,19 +102,19 @@ class Resque_Job_Status 'status' => $status, 'updated' => time(), ]; - Resque::redis()->set((string)$this, json_encode($statusPacket)); - // Expire the status for completed jobs after 24 hours - if (in_array($status, self::$completeStatuses)) { - Resque::redis()->expire((string)$this, 86400); - } + \Resque\Resque::redis()->set( + (string)$this, + json_encode($statusPacket), + ['ex' => \Resque\Redis::DEFAULT_REDIS_TTL], + ); } /** * Fetch the status for the job being monitored. * * @return mixed False if the status is not being monitored, otherwise the status as - * as an integer, based on the Resque_Job_Status constants. + * as an integer, based on the \Resque\Job\Status constants. */ public function get() { @@ -116,7 +122,7 @@ class Resque_Job_Status return false; } - $statusPacket = json_decode(Resque::redis()->get((string)$this), true); + $statusPacket = json_decode(\Resque\Resque::redis()->get((string)$this), true); if (!$statusPacket) { return false; } @@ -126,10 +132,12 @@ class Resque_Job_Status /** * Stop tracking the status of a job. + * + * @return void */ - public function stop() + public function stop(): void { - Resque::redis()->del((string)$this); + \Resque\Resque::redis()->del((string)$this); } /** diff --git a/lib/Resque/Log.php b/src/Resque/Log.php similarity index 73% rename from lib/Resque/Log.php rename to src/Resque/Log.php index 4182b23..2608d87 100644 --- a/lib/Resque/Log.php +++ b/src/Resque/Log.php @@ -1,18 +1,20 @@ + * @package Resque + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -class Resque_Log extends Psr\Log\AbstractLogger +class Log extends \Psr\Log\AbstractLogger { public $logLevel; - public function __construct($logLevel = "warning") + public function __construct($logLevel = 'warning') { $this->logLevel = strtolower($logLevel); } @@ -23,21 +25,22 @@ class Resque_Log extends Psr\Log\AbstractLogger * @param mixed $level PSR-3 log level constant, or equivalent string * @param string $message Message to log, may contain a { placeholder } * @param array $context Variables to replace { placeholder } + * * @return null */ - public function log($level, $message, array $context = []) + public function log($level, $message, array $context = []): void { $logLevels = [ - "emergency", - "alert", - "critical", - "error", - "warning", - "notice", - "info", - "debug" + 'emergency', + 'alert', + 'critical', + 'error', + 'warning', + 'notice', + 'info', + 'debug', ]; - + /** * Only log things with a higher level than the current log level. * e.g If set as 'alert' will only alert for 'emergency' and 'alert' logs. @@ -45,11 +48,11 @@ class Resque_Log extends Psr\Log\AbstractLogger if (array_search($level, $logLevels) <= array_search($this->logLevel, $logLevels)) { fwrite( STDOUT, - '[' . $level . '][' . strftime('%Y-%m-%d %T') . '] ' . $this->interpolate($message, $context) . PHP_EOL + '[' . $level . '][' . date('Y-m-d H:i:s') . '] ' . + $this->interpolate($message, $context) . PHP_EOL ); } return; - } /** diff --git a/lib/Resque/Redis.php b/src/Resque/Redis.php similarity index 84% rename from lib/Resque/Redis.php rename to src/Resque/Redis.php index ee21223..5537b55 100644 --- a/lib/Resque/Redis.php +++ b/src/Resque/Redis.php @@ -1,23 +1,27 @@ + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -class Resque_Redis +class Redis { /** * Redis Client - * @var Credis_Client + * + * @var \Credis_Client */ private $driver; /** * Redis namespace + * * @var string */ private static $defaultNamespace = 'resque:'; @@ -25,17 +29,22 @@ class Resque_Redis /** * A default host to connect to */ - const DEFAULT_HOST = 'localhost'; + public const DEFAULT_HOST = 'localhost'; /** * The default Redis port */ - const DEFAULT_PORT = 6379; + public const DEFAULT_PORT = 6379; /** * The default Redis Database number */ - const DEFAULT_DATABASE = 0; + public const DEFAULT_DATABASE = 0; + + /** + * Default Redis TTL (2 days) + */ + public const DEFAULT_REDIS_TTL = 172800; /** * @var array List of all commands in Redis that supply a key as their @@ -88,27 +97,20 @@ class Resque_Redis 'rename', 'rpoplpush' ]; - // sinterstore - // sunion - // sunionstore - // sdiff - // sdiffstore - // sinter - // smove - // mget - // msetnx - // mset - // renamenx /** * Set Redis namespace (prefix) default: resque + * * @param string $namespace + * + * @return void */ - public static function prefix($namespace) + public static function prefix(string $namespace): void { if (substr($namespace, -1) !== ':' && $namespace != '') { $namespace .= ':'; } + self::$defaultNamespace = $namespace; } @@ -116,8 +118,9 @@ class Resque_Redis * @param string|array $server A DSN or array * @param int $database A database number to select. However, if we find a valid database number in the DSN the * DSN-supplied value will be used instead and this parameter is ignored. - * @param object $client Optional Credis_Client instance instantiated by you - * @throws Resque_RedisException + * @param object $client Optional \Credis_Client instance instantiated by you + * + * @throws \Resque\RedisException */ public function __construct($server, $database = null, $client = null) { @@ -131,7 +134,7 @@ class Resque_Redis $timeout = isset($options['timeout']) ? intval($options['timeout']) : null; $persistent = isset($options['persistent']) ? $options['persistent'] : ''; $maxRetries = isset($options['max_connect_retries']) ? $options['max_connect_retries'] : 0; - $this->driver = new Credis_Client($host, $port, $timeout, $persistent); + $this->driver = new \Credis_Client($host, $port, $timeout, $persistent); $this->driver->setMaxConnectRetries($maxRetries); if ($password) { $this->driver->auth($password); @@ -145,8 +148,8 @@ class Resque_Redis if ($database !== null) { $this->driver->select($database); } - } catch (Exception $e) { - throw new Resque_RedisException('Error communicating with Redis: ' . $e->getMessage(), 0, $e); + } catch (\Exception $e) { + throw new RedisException('Error communicating with Redis: ' . $e->getMessage(), 0, $e); } } @@ -161,10 +164,11 @@ class Resque_Redis * Note: the 'user' part of the DSN is not used. * * @param string $dsn A DSN string + * * @return array An array of DSN compotnents, with 'false' values for any unknown components. e.g. * [host, port, db, user, pass, options] */ - public static function parseDsn($dsn) + public static function parseDsn($dsn): array { if ($dsn == '') { // Use a sensible default for an empty DNS string @@ -231,14 +235,16 @@ class Resque_Redis * * @param string $name The name of the method called. * @param array $args Array of supplied arguments to the method. + * * @return mixed Return value from Resident::call() based on the command. + * * @throws Resque_RedisException */ public function __call($name, $args) { if (in_array($name, $this->keyCommands)) { if (is_array($args[0])) { - foreach ($args[0] AS $i => $v) { + foreach ($args[0] as $i => $v) { $args[0][$i] = self::$defaultNamespace . $v; } } else { @@ -247,23 +253,18 @@ class Resque_Redis } try { return $this->driver->__call($name, $args); - } catch (CredisException $e) { - throw new Resque_RedisException('Error communicating with Redis: ' . $e->getMessage(), 0, $e); + } catch (\Exception $e) { + throw new RedisException('Error communicating with Redis: ' . $e->getMessage(), 0, $e); } } + /** + * Returns redis prefix + * + * @return string + */ public static function getPrefix(): string { return self::$defaultNamespace; } - - public static function removePrefix($string): string - { - $prefix = self::getPrefix(); - - if (substr($string, 0, strlen($prefix)) == $prefix) { - $string = substr($string, strlen($prefix), strlen($string)); - } - return $string; - } } diff --git a/src/Resque/RedisException.php b/src/Resque/RedisException.php new file mode 100644 index 0000000..c875887 --- /dev/null +++ b/src/Resque/RedisException.php @@ -0,0 +1,15 @@ + + * @license http://www.opensource.org/licenses/mit-license.php + */ + +class RedisException extends \Exception +{ +} diff --git a/lib/Resque.php b/src/Resque/Resque.php similarity index 84% rename from lib/Resque.php rename to src/Resque/Resque.php index a7bff07..f193e18 100644 --- a/lib/Resque.php +++ b/src/Resque/Resque.php @@ -1,18 +1,20 @@ + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ class Resque { - const VERSION = '1.4.3'; + public const VERSION = '2.5.3'; - const DEFAULT_INTERVAL = 5; + public const DEFAULT_INTERVAL = 5; /** * @var Resque_Redis Instance of Resque_Redis that talks to redis. @@ -50,19 +52,20 @@ class Resque /** * Return an instance of the Resque_Redis class instantiated for Resque. * - * @return Resque_Redis Instance of Resque_Redis. - * @throws Resque_RedisException + * @return \Resque\Redis Instance of Resque_Redis. + * + * @throws \Resque\RedisException */ public static function redis() { - if (self::$redis !== null) { + if (!is_null(self::$redis)) { return self::$redis; } if (is_callable(self::$redisServer)) { self::$redis = call_user_func(self::$redisServer, self::$redisDatabase); } else { - self::$redis = new Resque_Redis(self::$redisServer, self::$redisDatabase); + self::$redis = new \Resque\Redis(self::$redisServer, self::$redisDatabase); } return self::$redis; @@ -89,7 +92,7 @@ class Resque $pid = pcntl_fork(); if ($pid === -1) { - throw new RuntimeException('Unable to fork child worker.'); + throw new \RuntimeException('Unable to fork child worker.'); } return $pid; @@ -101,19 +104,23 @@ class Resque * * @param string $queue The name of the queue to add the job to. * @param array $item Job description as an array to be JSON encoded. + * * @return bool */ - public static function push($queue, $item) + public static function push($queue, $item): bool { $encodedItem = json_encode($item); if ($encodedItem === false) { return false; } + self::redis()->sadd('queues', $queue); + $length = self::redis()->rpush('queue:' . $queue, $encodedItem); if ($length < 1) { return false; } + return true; } @@ -122,6 +129,7 @@ class Resque * return it. * * @param string $queue The name of the queue to fetch an item from. + * * @return mixed Decoded item from the queue. */ public static function pop($queue) @@ -142,7 +150,7 @@ class Resque * @param array $items * @return integer number of deleted items */ - public static function dequeue($queue, $items = Array()) + public static function dequeue($queue, $items = []) { if (count($items) > 0) { return self::removeItems($queue, $items); @@ -155,13 +163,14 @@ class Resque * Remove specified queue * * @param string $queue The name of the queue to remove. - * @return integer Number of deleted items + * + * @return int Number of deleted items */ - public static function removeQueue($queue) + public static function removeQueue($queue): int { $num = self::removeList($queue); self::redis()->srem('queues', $queue); - return $num; + return intval($num); } /** @@ -170,12 +179,13 @@ class Resque * * @param array $queues * @param int $timeout + * * @return array|null|void */ public static function blpop(array $queues, $timeout) { - $list = array(); - foreach ($queues AS $queue) { + $list = []; + foreach ($queues as $queue) { $list[] = 'queue:' . $queue; } @@ -192,10 +202,10 @@ class Resque */ $queue = substr($item[0], strlen(self::redis()->getPrefix() . 'queue:')); - return array( + return [ 'queue' => $queue, 'payload' => json_decode($item[1], true) - ); + ]; } /** @@ -205,9 +215,9 @@ class Resque * * @return int The size of the queue. */ - public static function size($queue) + public static function size($queue): int { - return self::redis()->llen('queue:' . $queue); + return intval(self::redis()->llen('queue:' . $queue)); } /** @@ -223,20 +233,20 @@ class Resque public static function enqueue($queue, $class, $args = null, $trackStatus = false) { $id = Resque::generateJobId(); - $hookParams = array( + $hookParams = [ 'class' => $class, 'args' => $args, 'queue' => $queue, 'id' => $id, - ); + ]; try { - Resque_Event::trigger('beforeEnqueue', $hookParams); - } catch (Resque_Job_DontCreate $e) { + \Resque\Event::trigger('beforeEnqueue', $hookParams); + } catch (\Resque\Job\DontCreate $e) { return false; } - Resque_Job::create($queue, $class, $args, $trackStatus, $id); - Resque_Event::trigger('afterEnqueue', $hookParams); + \Resque\Job\Job::create($queue, $class, $args, $trackStatus, $id); + \Resque\Event::trigger('afterEnqueue', $hookParams); return $id; } @@ -245,11 +255,12 @@ class Resque * Reserve and return the next available job in the specified queue. * * @param string $queue Queue to fetch next available job from. - * @return false|object|Resque_Job + * + * @return \Resque\Job\Job|null */ - public static function reserve($queue) + public static function reserve($queue): ?\Resque\Job\Job { - return Resque_Job::reserve($queue); + return \Resque\Job\Job::reserve($queue); } /** @@ -257,13 +268,10 @@ class Resque * * @return array Array of queues. */ - public static function queues() + public static function queues(): array { $queues = self::redis()->smembers('queues'); - if (!is_array($queues)) { - $queues = array(); - } - return $queues; + return is_array($queues) ? $queues : []; } /** @@ -276,9 +284,10 @@ class Resque * * @param string $queue The name of the queue * @param array $items - * @return integer number of deleted items + * + * @return int number of deleted items */ - private static function removeItems($queue, $items = Array()) + private static function removeItems($queue, $items = []): int { $counter = 0; $originalQueue = 'queue:' . $queue; @@ -343,8 +352,11 @@ class Resque # class name with args , example: item[0] = ['class' => {'foo' => 1, 'bar' => 2}] } elseif (is_array($val)) { $decodedArgs = (array)$decoded['args'][0]; - if ($decoded['class'] == $key && - count($decodedArgs) > 0 && count(array_diff($decodedArgs, $val)) == 0) { + if ( + $decoded['class'] == $key + && count($decodedArgs) > 0 + && count(array_diff($decodedArgs, $val)) == 0 + ) { return true; } # class name with ID, example: item[0] = ['class' => 'id'] @@ -365,7 +377,7 @@ class Resque * @params string $queue the name of the queue * @param $queue * @return integer number of deleted items belongs to this list - * @throws Resque_RedisException + * @throws \Resque\RedisException */ private static function removeList($queue) { diff --git a/lib/Resque/Stat.php b/src/Resque/Stat.php similarity index 55% rename from lib/Resque/Stat.php rename to src/Resque/Stat.php index 0cca826..6cfa6c1 100644 --- a/lib/Resque/Stat.php +++ b/src/Resque/Stat.php @@ -1,22 +1,25 @@ + * @package Resque + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -class Resque_Stat +class Stat { /** * Get the value of the supplied statistic counter for the specified statistic. * * @param string $stat The name of the statistic to get the stats for. - * @return mixed Value of the statistic. + * + * @return int Value of the statistic. */ - public static function get($stat): int + public static function get(string $stat): int { return (int)Resque::redis()->get('stat:' . $stat); } @@ -26,11 +29,24 @@ class Resque_Stat * * @param string $stat The name of the statistic to increment. * @param int $by The amount to increment the statistic by. - * @return boolean True if successful, false if not. + * + * @return bool True if successful, false if not. */ - public static function incr($stat, $by = 1): bool + public static function incr(string $stat, int $by = 1): bool { - return (bool)Resque::redis()->incrby('stat:' . $stat, $by); + // Make sure we set a TTL by default + $set = Resque::redis()->set( + 'stat:' . $stat, + $by, + ['ex' => Redis::DEFAULT_REDIS_TTL, 'nx'], + ); + + // If it already exists, return the incrby value + if (!$set) { + return (bool)Resque::redis()->incrby('stat:' . $stat, $by); + } + + return true; } /** @@ -38,9 +54,10 @@ class Resque_Stat * * @param string $stat The name of the statistic to decrement. * @param int $by The amount to decrement the statistic by. - * @return boolean True if successful, false if not. + * + * @return bool True if successful, false if not. */ - public static function decr($stat, $by = 1): bool + public static function decr(string $stat, int $by = 1): bool { return (bool)Resque::redis()->decrby('stat:' . $stat, $by); } @@ -49,10 +66,11 @@ class Resque_Stat * Delete a statistic with the given name. * * @param string $stat The name of the statistic to delete. - * @return boolean True if successful, false if not. + * + * @return bool True if successful, false if not. */ - public static function clear($stat): bool + public static function clear(string $stat): bool { return (bool)Resque::redis()->del('stat:' . $stat); } -} \ No newline at end of file +} diff --git a/lib/Resque/Worker.php b/src/Resque/Worker.php similarity index 63% rename from lib/Resque/Worker.php rename to src/Resque/Worker.php index d4e385b..fc08303 100644 --- a/lib/Resque/Worker.php +++ b/src/Resque/Worker.php @@ -1,17 +1,20 @@ + * @package Resque + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -class Resque_Worker +class Worker { /** * @var LoggerInterface Logging object that impliments the PSR-3 LoggerInterface @@ -44,7 +47,7 @@ class Resque_Worker private $id; /** - * @var Resque_Job Current job, if any, being processed by this worker. + * @var \Resque\Job\Job Current job, if any, being processed by this worker. */ private $currentJob = null; @@ -66,7 +69,7 @@ class Resque_Worker */ public function __construct($queues) { - $this->logger = new Resque_Log(); + $this->logger = new Log(); if (!is_array($queues)) { $queues = [$queues]; @@ -80,6 +83,7 @@ class Resque_Worker /** * Return all workers known to Resque as instantiated instances. + * * @return array */ public static function all(): array @@ -93,14 +97,17 @@ class Resque_Worker foreach ($workers as $workerId) { $instances[] = self::find($workerId); } + return $instances; } /** * Given a worker ID, check if it is registered/valid. * - * @param string $workerId ID of the worker. - * @return boolean True if the worker exists, false if not. + * @param string $workerId ID of the worker + * + * @return boolean True if the worker exists, false if not + * * @throws Resque_RedisException */ public static function exists($workerId): bool @@ -111,20 +118,24 @@ class Resque_Worker /** * Given a worker ID, find it and return an instantiated worker class for it. * - * @param string $workerId The ID of the worker. - * @return bool|Resque_Worker + * @param string $workerId The ID of the worker + * + * @return Resque_Worker|bool + * * @throws Resque_RedisException */ public static function find($workerId) { - if (!self::exists($workerId) || false === strpos($workerId, ":")) { + if (false === strpos($workerId, ":") || !self::exists($workerId)) { return false; } + /** @noinspection PhpUnusedLocalVariableInspection */ list($hostname, $pid, $queues) = explode(':', $workerId, 3); $queues = explode(',', $queues); $worker = new self($queues); $worker->setId($workerId); + return $worker; } @@ -132,8 +143,10 @@ class Resque_Worker * Set the ID of this worker to a given ID string. * * @param string $workerId ID for the worker. + * + * @return void */ - public function setId($workerId) + public function setId($workerId): void { $this->id = $workerId; } @@ -146,9 +159,12 @@ class Resque_Worker * * @param int $interval How often to check for new jobs across the queues. * @param bool $blocking + * + * @return void + * * @throws Resque_RedisException */ - public function work($interval = Resque::DEFAULT_INTERVAL, $blocking = false) + public function work($interval = Resque::DEFAULT_INTERVAL, $blocking = false): void { $this->updateProcLine('Starting'); $this->startup(); @@ -162,8 +178,14 @@ class Resque_Worker $job = false; if (!$this->paused) { if ($blocking === true) { - $this->logger->log(Psr\Log\LogLevel::INFO, 'Starting blocking with timeout of {interval}', ['interval' => $interval]); - $this->updateProcLine('Waiting for ' . implode(',', $this->queues) . ' with blocking timeout ' . $interval); + $this->logger->log( + \Psr\Log\LogLevel::INFO, + 'Starting blocking with timeout of {interval}', + ['interval' => $interval], + ); + $this->updateProcLine( + 'Waiting for ' . implode(',', $this->queues) . ' with blocking timeout ' . $interval + ); } else { $this->updateProcLine('Waiting for ' . implode(',', $this->queues) . ' with interval ' . $interval); } @@ -179,7 +201,12 @@ class Resque_Worker if ($blocking === false) { // If no job was found, we sleep for $interval before continuing and checking again - $this->logger->log(Psr\Log\LogLevel::INFO, 'Sleeping for {interval}', ['interval' => $interval]); + $this->logger->log( + \Psr\Log\LogLevel::INFO, + 'Sleeping for {interval}', + ['interval' => $interval], + ); + if ($this->paused) { $this->updateProcLine('Paused'); } else { @@ -192,17 +219,19 @@ class Resque_Worker continue; } - $this->logger->log(Psr\Log\LogLevel::NOTICE, 'Starting work on {job}', ['job' => $job]); - Resque_Event::trigger('beforeFork', $job); + $this->logger->log(\Psr\Log\LogLevel::NOTICE, 'Starting work on {job}', ['job' => $job]); + Event::trigger('beforeFork', $job); $this->workingOn($job); $this->child = Resque::fork(); // Forked and we're the child. Run the job. if ($this->child === 0 || $this->child === false) { - $status = 'Processing ' . $job->queue . ' since ' . strftime('%F %T'); + $status = 'Processing ' . $job->queue + . ' (' . ($job->payload['class'] ?? '') . ') since ' + . date('Y-m-d H:i:s'); $this->updateProcLine($status); - $this->logger->log(Psr\Log\LogLevel::INFO, $status); + $this->logger->log(\Psr\Log\LogLevel::INFO, $status); /** @noinspection PhpParamsInspection */ $this->perform($job); if ($this->child === 0) { @@ -212,15 +241,15 @@ class Resque_Worker if ($this->child > 0) { // Parent process, sit and wait - $status = 'Forked ' . $this->child . ' at ' . strftime('%F %T'); + $status = 'Forked ' . $this->child . ' at ' . date('Y-m-d H:i:s'); $this->updateProcLine($status); - $this->logger->log(Psr\Log\LogLevel::INFO, $status); + $this->logger->log(\Psr\Log\LogLevel::INFO, $status); // Wait until the child process finishes before continuing pcntl_wait($status); $exitStatus = pcntl_wexitstatus($status); if ($exitStatus !== 0) { - $job->fail(new Resque_Job_DirtyExitException( + $job->fail(new \Resque\Job\DirtyExitException( 'Job exited with exit code ' . $exitStatus )); } @@ -234,29 +263,32 @@ class Resque_Worker } /** - * Process a single job. + * Process a single job * - * @param Resque_Job $job The job to be processed. + * @param \Resque\Job\Job $job The job to be processed + * + * @return void */ - public function perform(Resque_Job $job) + public function perform(\Resque\Job\Job $job): void { try { - Resque_Event::trigger('afterFork', $job); + Event::trigger('afterFork', $job); $job->perform(); - } catch (Exception $e) { - $this->logger->log(Psr\Log\LogLevel::CRITICAL, '{job} has failed {stack}', ['job' => $job, 'stack' => $e]); + } catch (\Exception $e) { + $this->logger->log(\Psr\Log\LogLevel::CRITICAL, '{job} has failed {stack}', ['job' => $job, 'stack' => $e]); $job->fail($e); return; } - $job->updateStatus(Resque_Job_Status::STATUS_COMPLETE); - $this->logger->log(Psr\Log\LogLevel::NOTICE, '{job} has finished', ['job' => $job]); + $job->updateStatus(\Resque\Job\Status::STATUS_COMPLETE); + $this->logger->log(\Psr\Log\LogLevel::NOTICE, '{job} has finished', ['job' => $job]); } /** - * @param bool $blocking - * @param int $timeout - * @return object|boolean Instance of Resque_Job if a job is found, false if not. + * @param bool $blocking + * @param int $timeout + * + * @return object|boolean - Instance of \Resque\Job\Job if a job is found, false if not */ public function reserve($blocking = false, $timeout = null) { @@ -266,17 +298,17 @@ class Resque_Worker } if ($blocking === true) { - $job = Resque_Job::reserveBlocking($queues, $timeout); - if ($job) { - $this->logger->log(Psr\Log\LogLevel::INFO, 'Found job on {queue}', ['queue' => $job->queue]); + $job = \Resque\Job\Job::reserveBlocking($queues, $timeout); + if (!is_null($job)) { + $this->logger->log(\Psr\Log\LogLevel::INFO, 'Found job on {queue}', ['queue' => $job->queue]); return $job; } } else { foreach ($queues as $queue) { - $this->logger->log(Psr\Log\LogLevel::INFO, 'Checking {queue} for jobs', ['queue' => $queue]); - $job = Resque_Job::reserve($queue); - if ($job) { - $this->logger->log(Psr\Log\LogLevel::INFO, 'Found job on {queue}', ['queue' => $job->queue]); + $this->logger->log(\Psr\Log\LogLevel::INFO, 'Checking {queue} for jobs', ['queue' => $queue]); + $job = \Resque\Job\Job::reserve($queue); + if (!is_null($job)) { + $this->logger->log(\Psr\Log\LogLevel::INFO, 'Found job on {queue}', ['queue' => $job->queue]); return $job; } } @@ -287,16 +319,17 @@ class Resque_Worker /** * Return an array containing all of the queues that this worker should use - * when searching for jobs. + * when searching for jobs * * If * is found in the list of queues, every queue will be searched in - * alphabetic order. (@see $fetch) + * alphabetic order. (@param boolean $fetch If true, and the queue is set to *, will fetch + * all queue names from redis * - * @param boolean $fetch If true, and the queue is set to *, will fetch - * all queue names from redis. - * @return array Array of associated queues. + * @param boolean $fetch + * + * @return array Array of associated queues */ - public function queues($fetch = true) + public function queues(bool $fetch = true): array { if (!in_array('*', $this->queues) || $fetch == false) { return $this->queues; @@ -304,17 +337,20 @@ class Resque_Worker $queues = Resque::queues(); sort($queues); + return $queues; } /** - * Perform necessary actions to start a worker. + * Perform necessary actions to start a worker + * + * @return void */ - private function startup() + private function startup(): void { $this->registerSigHandlers(); $this->pruneDeadWorkers(); - Resque_Event::trigger('beforeFirstFork', $this); + Event::trigger('beforeFirstFork', $this); $this->registerWorker(); } @@ -323,14 +359,16 @@ class Resque_Worker * the name of the currently running process to indicate the current state * of a worker. * - * @param string $status The updated process title. + * @param string $status The updated process title + * + * @return void */ - private function updateProcLine($status) + private function updateProcLine($status): void { $processTitle = 'resque-' . Resque::VERSION . ': ' . $status; if (function_exists('cli_set_process_title') && PHP_OS !== 'Darwin') { cli_set_process_title($processTitle); - } else if (function_exists('setproctitle')) { + } elseif (function_exists('setproctitle')) { setproctitle($processTitle); } } @@ -342,8 +380,10 @@ class Resque_Worker * INT: Shutdown immediately and stop processing jobs. * QUIT: Shutdown after the current job finishes processing. * USR1: Kill the forked child immediately and continue processing jobs. + * + * @return void */ - private function registerSigHandlers() + private function registerSigHandlers(): void { if (!function_exists('pcntl_signal')) { return; @@ -355,43 +395,51 @@ class Resque_Worker pcntl_signal(SIGUSR1, [$this, 'killChild']); pcntl_signal(SIGUSR2, [$this, 'pauseProcessing']); pcntl_signal(SIGCONT, [$this, 'unPauseProcessing']); - $this->logger->log(Psr\Log\LogLevel::DEBUG, 'Registered signals'); + $this->logger->log(\Psr\Log\LogLevel::DEBUG, 'Registered signals'); } /** - * Signal handler callback for USR2, pauses processing of new jobs. + * Signal handler callback for USR2, pauses processing of new jobs + * + * @return void */ - public function pauseProcessing() + public function pauseProcessing(): void { - $this->logger->log(Psr\Log\LogLevel::NOTICE, 'USR2 received; pausing job processing'); + $this->logger->log(\Psr\Log\LogLevel::NOTICE, 'USR2 received; pausing job processing'); $this->paused = true; } /** * Signal handler callback for CONT, resumes worker allowing it to pick * up new jobs. + * + * @return void */ - public function unPauseProcessing() + public function unPauseProcessing(): void { - $this->logger->log(Psr\Log\LogLevel::NOTICE, 'CONT received; resuming job processing'); + $this->logger->log(\Psr\Log\LogLevel::NOTICE, 'CONT received; resuming job processing'); $this->paused = false; } /** * Schedule a worker for shutdown. Will finish processing the current job * and when the timeout interval is reached, the worker will shut down. + * + * @return void */ - public function shutdown() + public function shutdown(): void { $this->shutdown = true; - $this->logger->log(Psr\Log\LogLevel::NOTICE, 'Shutting down'); + $this->logger->log(\Psr\Log\LogLevel::NOTICE, 'Shutting down'); } /** * Force an immediate shutdown of the worker, killing any child jobs * currently running. + * + * @return void */ - public function shutdownNow() + public function shutdownNow(): void { $this->shutdown(); $this->killChild(); @@ -400,21 +448,27 @@ class Resque_Worker /** * Kill a forked child job immediately. The job it is processing will not * be completed. + * + * @return void */ - public function killChild() + public function killChild(): void { if (!$this->child) { - $this->logger->log(Psr\Log\LogLevel::DEBUG, 'No child to kill.'); + $this->logger->log(\Psr\Log\LogLevel::DEBUG, 'No child to kill.'); return; } - $this->logger->log(Psr\Log\LogLevel::INFO, 'Killing child at {child}', ['child' => $this->child]); + $this->logger->log(\Psr\Log\LogLevel::INFO, 'Killing child at {child}', ['child' => $this->child]); if (exec('ps -o pid,state -p ' . $this->child, $output, $returnCode) && $returnCode != 1) { - $this->logger->log(Psr\Log\LogLevel::DEBUG, 'Child {child} found, killing.', ['child' => $this->child]); + $this->logger->log(\Psr\Log\LogLevel::DEBUG, 'Child {child} found, killing.', ['child' => $this->child]); posix_kill($this->child, SIGKILL); $this->child = null; } else { - $this->logger->log(Psr\Log\LogLevel::INFO, 'Child {child} not found, restarting.', ['child' => $this->child]); + $this->logger->log( + \Psr\Log\LogLevel::INFO, + 'Child {child} not found, restarting.', + ['child' => $this->child], + ); $this->shutdown(); } } @@ -426,8 +480,10 @@ class Resque_Worker * This is a form of garbage collection to handle cases where the * server may have been killed and the Resque workers did not die gracefully * and therefore leave state information in Redis. + * + * @return void */ - public function pruneDeadWorkers() + public function pruneDeadWorkers(): void { $workerPids = $this->workerPids(); $workers = self::all(); @@ -437,7 +493,11 @@ class Resque_Worker if ($host != $this->hostname || in_array($pid, $workerPids) || $pid == getmypid()) { continue; } - $this->logger->log(Psr\Log\LogLevel::INFO, 'Pruning dead worker: {worker}', ['worker' => (string)$worker]); + $this->logger->log( + \Psr\Log\LogLevel::INFO, + 'Pruning dead worker: {worker}', + ['worker' => (string)$worker], + ); $worker->unregisterWorker(); } } @@ -449,7 +509,7 @@ class Resque_Worker * * @return array Array of Resque worker process IDs. */ - public function workerPids() + public function workerPids(): array { $pids = []; exec('ps -A -o pid,command | grep [r]esque', $cmdOutput); @@ -461,100 +521,122 @@ class Resque_Worker /** * Register this worker in Redis. - * 48 hour TTL so we don't pollute the db on server termination. + * 48 hour TTL so we don't pollute the redis db on server termination. + * + * @return void */ - public function registerWorker() + public function registerWorker(): void { Resque::redis()->sadd('workers', (string)$this); - Resque::redis()->setex('worker:' . (string)$this . ':started', 172800, strftime('%a %b %d %H:%M:%S %Z %Y')); + Resque::redis()->set( + 'worker:' . (string)$this . ':started', + date('D M d H:i:s T Y'), + ['ex' => Redis::DEFAULT_REDIS_TTL], + ); } /** * Unregister this worker in Redis. (shutdown etc) + * + * @return void */ - public function unregisterWorker() + public function unregisterWorker(): void { if (is_object($this->currentJob)) { - $this->currentJob->fail(new Resque_Job_DirtyExitException); + $this->currentJob->fail(new \Resque\Job\DirtyExitException()); } $id = (string)$this; Resque::redis()->srem('workers', $id); Resque::redis()->del('worker:' . $id); Resque::redis()->del('worker:' . $id . ':started'); - Resque_Stat::clear('processed:' . $id); - Resque_Stat::clear('failed:' . $id); + Stat::clear('processed:' . $id); + Stat::clear('failed:' . $id); } /** - * Tell Redis which job we're currently working on. + * Tell Redis which job we're currently working on + * + * @param \Resque\Job\Job $job \Resque\Job\Job instance containing the job we're working on + * + * @return void * - * @param Resque_Job $job Resque_Job instance containing the job we're working on. * @throws Resque_RedisException */ - public function workingOn(Resque_Job $job) + public function workingOn(\Resque\Job\Job $job): void { $job->worker = $this; $this->currentJob = $job; - $job->updateStatus(Resque_Job_Status::STATUS_RUNNING); + $job->updateStatus(\Resque\Job\Status::STATUS_RUNNING); $data = json_encode([ 'queue' => $job->queue, - 'run_at' => strftime('%a %b %d %H:%M:%S %Z %Y'), + 'run_at' => date('D M d H:i:s T Y'), 'payload' => $job->payload ]); - Resque::redis()->set('worker:' . $job->worker, $data); + + Resque::redis()->set( + 'worker:' . $job->worker, + $data, + ['ex' => Redis::DEFAULT_REDIS_TTL], + ); } /** * Notify Redis that we've finished working on a job, clearing the working - * state and incrementing the job stats. + * state and incrementing the job stats + * + * @return void */ - public function doneWorking() + public function doneWorking(): void { $this->currentJob = null; - Resque_Stat::incr('processed'); - Resque_Stat::incr('processed:' . (string)$this); + Stat::incr('processed'); + Stat::incr('processed:' . (string)$this); Resque::redis()->del('worker:' . (string)$this); } /** - * Generate a string representation of this worker. + * Generate a string representation of this worker * - * @return string String identifier for this worker instance. + * @return string String identifier for this worker instance */ - public function __toString() + public function __toString(): string { - return $this->id; + return (string) $this->id; } /** - * Return an object describing the job this worker is currently working on. + * Return an object describing the job this worker is currently working on * - * @return array Array with details of current job. + * @return array Array with details of current job */ public function job(): array { $job = Resque::redis()->get('worker:' . $this); + return $job ? json_decode($job, true) : []; } /** - * Get a statistic belonging to this worker. + * Get a statistic belonging to this worker * * @param string $stat Statistic to fetch. + * * @return int Statistic value. */ - public function getStat($stat) + public function getStat(string $stat): int { - return Resque_Stat::get($stat . ':' . $this); + return \Resque\Stat::get($stat . ':' . $this); } /** * Inject the logging object into the worker * - * @param Psr\Log\LoggerInterface $logger + * @param \Psr\Log\LoggerInterface $logger + * + * @return void */ - public function setLogger(Psr\Log\LoggerInterface $logger) + public function setLogger(\Psr\Log\LoggerInterface $logger): void { $this->logger = $logger; } diff --git a/test/Resque/Tests/JobStatusTest.php b/test/Resque/Tests/JobStatusTest.php deleted file mode 100644 index b2120ba..0000000 --- a/test/Resque/Tests/JobStatusTest.php +++ /dev/null @@ -1,109 +0,0 @@ - - * @license http://www.opensource.org/licenses/mit-license.php - */ - -class Resque_Tests_JobStatusTest extends Resque_Tests_TestCase -{ - /** - * @var \Resque_Worker - */ - protected $worker; - - public function setUp() - { - parent::setUp(); - - // Register a worker to test with - $this->worker = new Resque_Worker('jobs'); - $this->worker->setLogger(new Resque_Log()); - } - - public function testJobStatusCanBeTracked() - { - $token = Resque::enqueue('jobs', 'Test_Job', null, true); - $status = new Resque_Job_Status($token); - $this->assertTrue($status->isTracking()); - } - - public function testJobStatusIsReturnedViaJobInstance() - { - Resque::enqueue('jobs', 'Test_Job', null, true); - $job = Resque_Job::reserve('jobs'); - $this->assertEquals(Resque_Job_Status::STATUS_WAITING, $job->getStatus()); - - } - - public function testQueuedJobReturnsQueuedStatus() - { - $token = Resque::enqueue('jobs', 'Test_Job', null, true); - $status = new Resque_Job_Status($token); - $this->assertEquals(Resque_Job_Status::STATUS_WAITING, $status->get()); - } - - public function testRunningJobReturnsRunningStatus() - { - $token = Resque::enqueue('jobs', 'Failing_Job', null, true); - $job = $this->worker->reserve(); - $this->worker->workingOn($job); - $status = new Resque_Job_Status($token); - $this->assertEquals(Resque_Job_Status::STATUS_RUNNING, $status->get()); - } - - public function testFailedJobReturnsFailedStatus() - { - $token = Resque::enqueue('jobs', 'Failing_Job', null, true); - $this->worker->work(0); - $status = new Resque_Job_Status($token); - $this->assertEquals(Resque_Job_Status::STATUS_FAILED, $status->get()); - } - - public function testCompletedJobReturnsCompletedStatus() - { - $token = Resque::enqueue('jobs', 'Test_Job', null, true); - $this->worker->work(0); - $status = new Resque_Job_Status($token); - $this->assertEquals(Resque_Job_Status::STATUS_COMPLETE, $status->get()); - } - - public function testStatusIsNotTrackedWhenToldNotTo() - { - $token = Resque::enqueue('jobs', 'Test_Job', null, false); - $status = new Resque_Job_Status($token); - $this->assertFalse($status->isTracking()); - } - - public function testStatusTrackingCanBeStopped() - { - Resque_Job_Status::create('test'); - $status = new Resque_Job_Status('test'); - $this->assertEquals(Resque_Job_Status::STATUS_WAITING, $status->get()); - $status->stop(); - $this->assertFalse($status->get()); - } - - public function testRecreatedJobWithTrackingStillTracksStatus() - { - $originalToken = Resque::enqueue('jobs', 'Test_Job', null, true); - $job = $this->worker->reserve(); - - // Mark this job as being worked on to ensure that the new status is still - // waiting. - $this->worker->workingOn($job); - - // Now recreate it - $newToken = $job->recreate(); - - // Make sure we've got a new job returned - $this->assertNotEquals($originalToken, $newToken); - - // Now check the status of the new job - $newJob = Resque_Job::reserve('jobs'); - $this->assertEquals(Resque_Job_Status::STATUS_WAITING, $newJob->getStatus()); - } -} \ No newline at end of file diff --git a/test/Resque/Tests/JobTest.php b/test/Resque/Tests/JobTest.php deleted file mode 100644 index ada26cb..0000000 --- a/test/Resque/Tests/JobTest.php +++ /dev/null @@ -1,417 +0,0 @@ - - * @license http://www.opensource.org/licenses/mit-license.php - */ - -class Resque_Tests_JobTest extends Resque_Tests_TestCase -{ - protected $worker; - - public function setUp() - { - parent::setUp(); - - // Register a worker to test with - $this->worker = new Resque_Worker('jobs'); - $this->worker->setLogger(new Resque_Log()); - $this->worker->registerWorker(); - } - - public function testJobCanBeQueued() - { - $this->assertTrue((bool)Resque::enqueue('jobs', 'Test_Job')); - } - - public function testQeueuedJobCanBeReserved() - { - Resque::enqueue('jobs', 'Test_Job'); - - $job = Resque_Job::reserve('jobs'); - if ($job == false) { - $this->fail('Job could not be reserved.'); - } - $this->assertEquals('jobs', $job->queue); - $this->assertEquals('Test_Job', $job->payload['class']); - } - - /** - * @expectedException InvalidArgumentException - */ - public function testObjectArgumentsCannotBePassedToJob() - { - $args = new stdClass; - $args->test = 'somevalue'; - Resque::enqueue('jobs', 'Test_Job', $args); - } - - public function testQueuedJobReturnsExactSamePassedInArguments() - { - $args = [ - 'int' => 123, - 'numArray' => [ - 1, - 2, - ], - 'assocArray' => [ - 'key1' => 'value1', - 'key2' => 'value2' - ], - ]; - Resque::enqueue('jobs', 'Test_Job', $args); - $job = Resque_Job::reserve('jobs'); - - $this->assertEquals($args, $job->getArguments()); - } - - public function testAfterJobIsReservedItIsRemoved() - { - Resque::enqueue('jobs', 'Test_Job'); - Resque_Job::reserve('jobs'); - $this->assertFalse(Resque_Job::reserve('jobs')); - } - - public function testRecreatedJobMatchesExistingJob() - { - $args = [ - 'int' => 123, - 'numArray' => [ - 1, - 2, - ], - 'assocArray' => [ - 'key1' => 'value1', - 'key2' => 'value2' - ], - ]; - - Resque::enqueue('jobs', 'Test_Job', $args); - $job = Resque_Job::reserve('jobs'); - - // Now recreate it - $job->recreate(); - - $newJob = Resque_Job::reserve('jobs'); - $this->assertEquals($job->payload['class'], $newJob->payload['class']); - $this->assertEquals($job->getArguments(), $newJob->getArguments()); - } - - - public function testFailedJobExceptionsAreCaught() - { - $payload = [ - 'class' => 'Failing_Job', - 'args' => null - ]; - $job = new Resque_Job('jobs', $payload); - $job->worker = $this->worker; - - $this->worker->perform($job); - - $this->assertEquals(1, Resque_Stat::get('failed')); - $this->assertEquals(1, Resque_Stat::get('failed:' . $this->worker)); - } - - /** - * @expectedException Resque_Exception - */ - public function testJobWithoutPerformMethodThrowsException() - { - Resque::enqueue('jobs', 'Test_Job_Without_Perform_Method'); - $job = $this->worker->reserve(); - $job->worker = $this->worker; - $job->perform(); - } - - /** - * @expectedException Resque_Exception - */ - public function testInvalidJobThrowsException() - { - Resque::enqueue('jobs', 'Invalid_Job'); - $job = $this->worker->reserve(); - $job->worker = $this->worker; - $job->perform(); - } - - public function testJobWithSetUpCallbackFiresSetUp() - { - $payload = [ - 'class' => 'Test_Job_With_SetUp', - 'args' => [ - 'somevar', - 'somevar2', - ], - ]; - $job = new Resque_Job('jobs', $payload); - $job->perform(); - - $this->assertTrue(Test_Job_With_SetUp::$called); - } - - public function testJobWithTearDownCallbackFiresTearDown() - { - $payload = [ - 'class' => 'Test_Job_With_TearDown', - 'args' => [ - 'somevar', - 'somevar2', - ], - ]; - $job = new Resque_Job('jobs', $payload); - $job->perform(); - - $this->assertTrue(Test_Job_With_TearDown::$called); - } - - public function testNamespaceNaming() - { - $fixture = [ - ['test' => 'more:than:one:with:', 'assertValue' => 'more:than:one:with:'], - ['test' => 'more:than:one:without', 'assertValue' => 'more:than:one:without:'], - ['test' => 'resque', 'assertValue' => 'resque:'], - ['test' => 'resque:', 'assertValue' => 'resque:'], - ]; - - foreach ($fixture as $item) { - Resque_Redis::prefix($item['test']); - $this->assertEquals(Resque_Redis::getPrefix(), $item['assertValue']); - } - } - - public function testJobWithNamespace() - { - Resque_Redis::prefix('php'); - $queue = 'jobs'; - $payload = ['another_value']; - Resque::enqueue($queue, 'Test_Job_With_TearDown', $payload); - - $this->assertEquals(Resque::queues(), ['jobs']); - $this->assertEquals(Resque::size($queue), 1); - - Resque_Redis::prefix('resque'); - $this->assertEquals(Resque::size($queue), 0); - } - - public function testDequeueAll() - { - $queue = 'jobs'; - Resque::enqueue($queue, 'Test_Job_Dequeue'); - Resque::enqueue($queue, 'Test_Job_Dequeue'); - $this->assertEquals(Resque::size($queue), 2); - $this->assertEquals(Resque::dequeue($queue), 2); - $this->assertEquals(Resque::size($queue), 0); - } - - public function testDequeueMakeSureNotDeleteOthers() - { - $queue = 'jobs'; - Resque::enqueue($queue, 'Test_Job_Dequeue'); - Resque::enqueue($queue, 'Test_Job_Dequeue'); - $other_queue = 'other_jobs'; - Resque::enqueue($other_queue, 'Test_Job_Dequeue'); - Resque::enqueue($other_queue, 'Test_Job_Dequeue'); - $this->assertEquals(Resque::size($queue), 2); - $this->assertEquals(Resque::size($other_queue), 2); - $this->assertEquals(Resque::dequeue($queue), 2); - $this->assertEquals(Resque::size($queue), 0); - $this->assertEquals(Resque::size($other_queue), 2); - } - - public function testDequeueSpecificItem() - { - $queue = 'jobs'; - Resque::enqueue($queue, 'Test_Job_Dequeue1'); - Resque::enqueue($queue, 'Test_Job_Dequeue2'); - $this->assertEquals(Resque::size($queue), 2); - $test = ['Test_Job_Dequeue2']; - $this->assertEquals(Resque::dequeue($queue, $test), 1); - $this->assertEquals(Resque::size($queue), 1); - } - - public function testDequeueSpecificMultipleItems() - { - $queue = 'jobs'; - Resque::enqueue($queue, 'Test_Job_Dequeue1'); - Resque::enqueue($queue, 'Test_Job_Dequeue2'); - Resque::enqueue($queue, 'Test_Job_Dequeue3'); - $this->assertEquals(Resque::size($queue), 3); - $test = ['Test_Job_Dequeue2', 'Test_Job_Dequeue3']; - $this->assertEquals(Resque::dequeue($queue, $test), 2); - $this->assertEquals(Resque::size($queue), 1); - } - - public function testDequeueNonExistingItem() - { - $queue = 'jobs'; - Resque::enqueue($queue, 'Test_Job_Dequeue1'); - Resque::enqueue($queue, 'Test_Job_Dequeue2'); - Resque::enqueue($queue, 'Test_Job_Dequeue3'); - $this->assertEquals(Resque::size($queue), 3); - $test = ['Test_Job_Dequeue4']; - $this->assertEquals(Resque::dequeue($queue, $test), 0); - $this->assertEquals(Resque::size($queue), 3); - } - - public function testDequeueNonExistingItem2() - { - $queue = 'jobs'; - Resque::enqueue($queue, 'Test_Job_Dequeue1'); - Resque::enqueue($queue, 'Test_Job_Dequeue2'); - Resque::enqueue($queue, 'Test_Job_Dequeue3'); - $this->assertEquals(Resque::size($queue), 3); - $test = ['Test_Job_Dequeue4', 'Test_Job_Dequeue1']; - $this->assertEquals(Resque::dequeue($queue, $test), 1); - $this->assertEquals(Resque::size($queue), 2); - } - - public function testDequeueItemID() - { - $queue = 'jobs'; - Resque::enqueue($queue, 'Test_Job_Dequeue'); - $qid = Resque::enqueue($queue, 'Test_Job_Dequeue'); - $this->assertEquals(Resque::size($queue), 2); - $test = ['Test_Job_Dequeue' => $qid]; - $this->assertEquals(Resque::dequeue($queue, $test), 1); - $this->assertEquals(Resque::size($queue), 1); - } - - public function testDequeueWrongItemID() - { - $queue = 'jobs'; - Resque::enqueue($queue, 'Test_Job_Dequeue'); - $qid = Resque::enqueue($queue, 'Test_Job_Dequeue'); - $this->assertEquals(Resque::size($queue), 2); - #qid right but class name is wrong - $test = ['Test_Job_Dequeue1' => $qid]; - $this->assertEquals(Resque::dequeue($queue, $test), 0); - $this->assertEquals(Resque::size($queue), 2); - } - - public function testDequeueWrongItemID2() - { - $queue = 'jobs'; - Resque::enqueue($queue, 'Test_Job_Dequeue'); - Resque::enqueue($queue, 'Test_Job_Dequeue'); - $this->assertEquals(Resque::size($queue), 2); - $test = ['Test_Job_Dequeue' => 'r4nD0mH4sh3dId']; - $this->assertEquals(Resque::dequeue($queue, $test), 0); - $this->assertEquals(Resque::size($queue), 2); - } - - public function testDequeueItemWithArg() - { - $queue = 'jobs'; - $arg = ['foo' => 1, 'bar' => 2]; - Resque::enqueue($queue, 'Test_Job_Dequeue9'); - Resque::enqueue($queue, 'Test_Job_Dequeue9', $arg); - $this->assertEquals(Resque::size($queue), 2); - $test = ['Test_Job_Dequeue9' => $arg]; - $this->assertEquals(Resque::dequeue($queue, $test), 1); - #$this->assertEquals(Resque::size($queue), 1); - } - - public function testDequeueSeveralItemsWithArgs() - { - // GIVEN - $queue = 'jobs'; - $args = ['foo' => 1, 'bar' => 10]; - $removeArgs = ['foo' => 1, 'bar' => 2]; - Resque::enqueue($queue, 'Test_Job_Dequeue9', $args); - Resque::enqueue($queue, 'Test_Job_Dequeue9', $removeArgs); - Resque::enqueue($queue, 'Test_Job_Dequeue9', $removeArgs); - $this->assertEquals(Resque::size($queue), 3, "Failed to add 3 items."); - - // WHEN - $test = ['Test_Job_Dequeue9' => $removeArgs]; - $removedItems = Resque::dequeue($queue, $test); - - // THEN - $this->assertEquals($removedItems, 2); - $this->assertEquals(Resque::size($queue), 1); - $item = Resque::pop($queue); - $this->assertInternalType('array', $item['args']); - $this->assertEquals(10, $item['args'][0]['bar'], 'Wrong items were dequeued from queue!'); - } - - public function testDequeueItemWithUnorderedArg() - { - $queue = 'jobs'; - $arg = ['foo' => 1, 'bar' => 2]; - $arg2 = ['bar' => 2, 'foo' => 1]; - Resque::enqueue($queue, 'Test_Job_Dequeue'); - Resque::enqueue($queue, 'Test_Job_Dequeue', $arg); - $this->assertEquals(Resque::size($queue), 2); - $test = ['Test_Job_Dequeue' => $arg2]; - $this->assertEquals(Resque::dequeue($queue, $test), 1); - $this->assertEquals(Resque::size($queue), 1); - } - - public function testDequeueItemWithiWrongArg() - { - $queue = 'jobs'; - $arg = ['foo' => 1, 'bar' => 2]; - $arg2 = ['foo' => 2, 'bar' => 3]; - Resque::enqueue($queue, 'Test_Job_Dequeue'); - Resque::enqueue($queue, 'Test_Job_Dequeue', $arg); - $this->assertEquals(Resque::size($queue), 2); - $test = ['Test_Job_Dequeue' => $arg2]; - $this->assertEquals(Resque::dequeue($queue, $test), 0); - $this->assertEquals(Resque::size($queue), 2); - } - - public function testUseDefaultFactoryToGetJobInstance() - { - $payload = [ - 'class' => 'Some_Job_Class', - 'args' => null - ]; - $job = new Resque_Job('jobs', $payload); - $instance = $job->getInstance(); - $this->assertInstanceOf('Some_Job_Class', $instance); - } - - public function testUseFactoryToGetJobInstance() - { - $payload = [ - 'class' => 'Some_Job_Class', - 'args' => [[]] - ]; - $job = new Resque_Job('jobs', $payload); - $factory = new Some_Stub_Factory(); - $job->setJobFactory($factory); - $instance = $job->getInstance(); - $this->assertInstanceOf('Resque_JobInterface', $instance); - } -} - -class Some_Job_Class implements Resque_JobInterface -{ - - /** - * @return bool - */ - public function perform() - { - return true; - } -} - -class Some_Stub_Factory implements Resque_Job_FactoryInterface -{ - - /** - * @param $className - * @param array $args - * @param $queue - * @return Resque_JobInterface - */ - public function create($className, $args, $queue) - { - return new Some_Job_Class(); - } -} diff --git a/test/Resque/Tests/EventTest.php b/tests/Resque/Tests/EventTest.php similarity index 73% rename from test/Resque/Tests/EventTest.php rename to tests/Resque/Tests/EventTest.php index 24a83c7..9e9346f 100644 --- a/test/Resque/Tests/EventTest.php +++ b/tests/Resque/Tests/EventTest.php @@ -1,43 +1,45 @@ + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -class Resque_Tests_EventTest extends Resque_Tests_TestCase +class EventTest extends TestCase { private $callbacksHit = []; private $worker; - public function setUp() + public function setUp(): void { - Test_Job::$called = false; + TestJob::$called = false; // Register a worker to test with - $this->worker = new Resque_Worker('jobs'); - $this->worker->setLogger(new Resque_Log()); + $this->worker = new \Resque\Worker('jobs'); + $this->worker->setLogger(new \Resque\Log()); $this->worker->registerWorker(); } - public function tearDown() + public function tearDown(): void { - Resque_Event::clearListeners(); + \Resque\Event::clearListeners(); $this->callbacksHit = []; } public function getEventTestJob() { $payload = [ - 'class' => 'Test_Job', + 'class' => '\Resque\Test\TestJob', 'args' => [ ['somevar'], ], ]; - $job = new Resque_Job('jobs', $payload); + $job = new \Resque\Job\Job('jobs', $payload); $job->worker = $this->worker; return $job; } @@ -58,7 +60,7 @@ class Resque_Tests_EventTest extends Resque_Tests_TestCase */ public function testEventCallbacksFire($event, $callback) { - Resque_Event::listen($event, [$this, $callback]); + \Resque\Event::listen($event, [$this, $callback]); $job = $this->getEventTestJob(); $this->worker->perform($job); @@ -72,8 +74,8 @@ class Resque_Tests_EventTest extends Resque_Tests_TestCase $event = 'beforeFork'; $callback = 'beforeForkEventCallback'; - Resque_Event::listen($event, [$this, $callback]); - Resque::enqueue('jobs', 'Test_Job', [ + \Resque\Event::listen($event, [$this, $callback]); + \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob', [ 'somevar' ]); $this->getEventTestJob(); @@ -86,8 +88,8 @@ class Resque_Tests_EventTest extends Resque_Tests_TestCase $event = 'beforeEnqueue'; $callback = 'beforeEnqueueEventCallback'; - Resque_Event::listen($event, [$this, $callback]); - Resque::enqueue('jobs', 'Test_Job', [ + \Resque\Event::listen($event, [$this, $callback]); + \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob', [ 'somevar' ]); $this->assertContains($callback, $this->callbacksHit, $event . ' callback (' . $callback . ') was not called'); @@ -96,22 +98,22 @@ class Resque_Tests_EventTest extends Resque_Tests_TestCase public function testBeforePerformEventCanStopWork() { $callback = 'beforePerformEventDontPerformCallback'; - Resque_Event::listen('beforePerform', [$this, $callback]); + \Resque\Event::listen('beforePerform', [$this, $callback]); $job = $this->getEventTestJob(); $this->assertFalse($job->perform()); $this->assertContains($callback, $this->callbacksHit, $callback . ' callback was not called'); - $this->assertFalse(Test_Job::$called, 'Job was still performed though Resque_Job_DontPerform was thrown'); + $this->assertFalse(TestJob::$called, 'Job was still performed though Resque_Job_DontPerform was thrown'); } public function testBeforeEnqueueEventStopsJobCreation() { $callback = 'beforeEnqueueEventDontCreateCallback'; - Resque_Event::listen('beforeEnqueue', [$this, $callback]); - Resque_Event::listen('afterEnqueue', [$this, 'afterEnqueueEventCallback']); + \Resque\Event::listen('beforeEnqueue', [$this, $callback]); + \Resque\Event::listen('afterEnqueue', [$this, 'afterEnqueueEventCallback']); - $result = Resque::enqueue('test_job', 'TestClass'); + $result = \Resque\Resque::enqueue('jobs', '\Resque\Test\TestClass'); $this->assertContains($callback, $this->callbacksHit, $callback . ' callback was not called'); $this->assertNotContains('afterEnqueueEventCallback', $this->callbacksHit, 'afterEnqueue was still called, even though it should not have been'); $this->assertFalse($result); @@ -122,8 +124,8 @@ class Resque_Tests_EventTest extends Resque_Tests_TestCase $callback = 'afterEnqueueEventCallback'; $event = 'afterEnqueue'; - Resque_Event::listen($event, [$this, $callback]); - Resque::enqueue('jobs', 'Test_Job', [ + \Resque\Event::listen($event, [$this, $callback]); + \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob', [ 'somevar' ]); $this->assertContains($callback, $this->callbacksHit, $event . ' callback (' . $callback . ') was not called'); @@ -134,8 +136,8 @@ class Resque_Tests_EventTest extends Resque_Tests_TestCase $callback = 'beforePerformEventCallback'; $event = 'beforePerform'; - Resque_Event::listen($event, [$this, $callback]); - Resque_Event::stopListening($event, [$this, $callback]); + \Resque\Event::listen($event, [$this, $callback]); + \Resque\Event::stopListening($event, [$this, $callback]); $job = $this->getEventTestJob(); $this->worker->perform($job); @@ -149,20 +151,20 @@ class Resque_Tests_EventTest extends Resque_Tests_TestCase public function beforePerformEventDontPerformCallback() { $this->callbacksHit[] = __FUNCTION__; - throw new Resque_Job_DontPerform(); + throw new \Resque\Job\DontPerform(); } public function beforeEnqueueEventDontCreateCallback() { $this->callbacksHit[] = __FUNCTION__; - throw new Resque_Job_DontCreate(); + throw new \Resque\Job\DontCreate(); } public function assertValidEventCallback($function, $job) { $this->callbacksHit[] = $function; - if (!$job instanceof Resque_Job) { - $this->fail('Callback job argument is not an instance of Resque_Job'); + if (!$job instanceof \Resque\Job\Job) { + $this->fail('Callback job argument is not an instance of \Resque\Job\Job'); } $args = $job->getArguments(); $this->assertEquals($args[0], 'somevar'); @@ -171,7 +173,7 @@ class Resque_Tests_EventTest extends Resque_Tests_TestCase public function afterEnqueueEventCallback($class, $args) { $this->callbacksHit[] = __FUNCTION__; - $this->assertEquals('Test_Job', $class); + $this->assertEquals('\Resque\Test\TestJob', $class); $this->assertEquals([ 'somevar', ], $args); diff --git a/tests/Resque/Tests/JobStatusTest.php b/tests/Resque/Tests/JobStatusTest.php new file mode 100644 index 0000000..a7185cc --- /dev/null +++ b/tests/Resque/Tests/JobStatusTest.php @@ -0,0 +1,110 @@ + + * @license http://www.opensource.org/licenses/mit-license.php + */ + +class JobStatusTest extends TestCase +{ + /** + * @var \Resque\Worker + */ + protected $worker; + + public function setUp(): void + { + parent::setUp(); + + // Register a worker to test with + $this->worker = new \Resque\Worker('jobs'); + $this->worker->setLogger(new \Resque\Log()); + } + + public function testJobStatusCanBeTracked() + { + $token = \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob', null, true); + $status = new \Resque\Job\Status($token); + $this->assertTrue($status->isTracking()); + } + + public function testJobStatusIsReturnedViaJobInstance() + { + \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob', null, true); + $job = \Resque\Job\Job::reserve('jobs'); + $this->assertEquals(\Resque\Job\Status::STATUS_WAITING, $job->getStatus()); + } + + public function testQueuedJobReturnsQueuedStatus() + { + $token = \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob', null, true); + $status = new \Resque\Job\Status($token); + $this->assertEquals(\Resque\Job\Status::STATUS_WAITING, $status->get()); + } + + public function testRunningJobReturnsRunningStatus() + { + $token = \Resque\Resque::enqueue('jobs', '\Resque\Test\FailingJob', null, true); + $job = $this->worker->reserve(); + $this->worker->workingOn($job); + $status = new \Resque\Job\Status($token); + $this->assertEquals(\Resque\Job\Status::STATUS_RUNNING, $status->get()); + } + + public function testFailedJobReturnsFailedStatus() + { + $token = \Resque\Resque::enqueue('jobs', '\Resque\Test\FailingJob', null, true); + $this->worker->work(0); + $status = new \Resque\Job\Status($token); + $this->assertEquals(\Resque\Job\Status::STATUS_FAILED, $status->get()); + } + + public function testCompletedJobReturnsCompletedStatus() + { + $token = \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob', null, true); + $this->worker->work(0); + $status = new \Resque\Job\Status($token); + $this->assertEquals(\Resque\Job\Status::STATUS_COMPLETE, $status->get()); + } + + public function testStatusIsNotTrackedWhenToldNotTo() + { + $token = \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob', null, false); + $status = new \Resque\Job\Status($token); + $this->assertFalse($status->isTracking()); + } + + public function testStatusTrackingCanBeStopped() + { + \Resque\Job\Status::create('test'); + $status = new \Resque\Job\Status('test'); + $this->assertEquals(\Resque\Job\Status::STATUS_WAITING, $status->get()); + $status->stop(); + $this->assertFalse($status->get()); + } + + public function testRecreatedJobWithTrackingStillTracksStatus() + { + $originalToken = \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob', null, true); + $job = $this->worker->reserve(); + + // Mark this job as being worked on to ensure that the new status is still + // waiting. + $this->worker->workingOn($job); + + // Now recreate it + $newToken = $job->recreate(); + + // Make sure we've got a new job returned + $this->assertNotEquals($originalToken, $newToken); + + // Now check the status of the new job + $newJob = \Resque\Job\Job::reserve('jobs'); + $this->assertEquals(\Resque\Job\Status::STATUS_WAITING, $newJob->getStatus()); + } +} \ No newline at end of file diff --git a/tests/Resque/Tests/JobTest.php b/tests/Resque/Tests/JobTest.php new file mode 100644 index 0000000..f6c261f --- /dev/null +++ b/tests/Resque/Tests/JobTest.php @@ -0,0 +1,431 @@ + + * @license http://www.opensource.org/licenses/mit-license.php + */ + +class JobTest extends TestCase +{ + protected $worker; + + public function setUp(): void + { + parent::setUp(); + + // Register a worker to test with + $this->worker = new \Resque\Worker('jobs'); + $this->worker->setLogger(new \Resque\Log()); + $this->worker->registerWorker(); + } + + public function testJobCanBeQueued() + { + $this->assertTrue((bool)\Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob')); + } + + public function testQeueuedJobCanBeReserved() + { + \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob'); + + $job = \Resque\Job\Job::reserve('jobs'); + if (is_null($job)) { + $this->fail('Job could not be reserved.'); + } + $this->assertEquals('jobs', $job->queue); + $this->assertEquals('\Resque\Test\TestJob', $job->payload['class']); + } + + /** + * @expectedException \InvalidArgumentException + */ + public function testObjectArgumentsCannotBePassedToJob() + { + $this->expectException(\InvalidArgumentException::class); + + $args = new \stdClass(); + $args->test = 'somevalue'; + \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob', $args); + } + + public function testQueuedJobReturnsExactSamePassedInArguments() + { + $args = [ + 'int' => 123, + 'numArray' => [ + 1, + 2, + ], + 'assocArray' => [ + 'key1' => 'value1', + 'key2' => 'value2' + ], + ]; + \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob', $args); + $job = \Resque\Job\Job::reserve('jobs'); + + $this->assertEquals($args, $job->getArguments()); + } + + public function testAfterJobIsReservedItIsRemoved() + { + \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob'); + \Resque\Job\Job::reserve('jobs'); + $this->assertNull(\Resque\Job\Job::reserve('jobs')); + } + + public function testRecreatedJobMatchesExistingJob() + { + $args = [ + 'int' => 123, + 'numArray' => [ + 1, + 2, + ], + 'assocArray' => [ + 'key1' => 'value1', + 'key2' => 'value2' + ], + ]; + + \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob', $args); + $job = \Resque\Job\Job::reserve('jobs'); + + // Now recreate it + $job->recreate(); + + $newJob = \Resque\Job\Job::reserve('jobs'); + $this->assertEquals($job->payload['class'], $newJob->payload['class']); + $this->assertEquals($job->getArguments(), $newJob->getArguments()); + } + + + public function testFailedJobExceptionsAreCaught() + { + $payload = [ + 'class' => '\Resque\Test\FailingJob', + 'args' => null + ]; + $job = new \Resque\Job\Job('jobs', $payload); + $job->worker = $this->worker; + + $this->worker->perform($job); + + $this->assertEquals(1, \Resque\Stat::get('failed')); + $this->assertEquals(1, \Resque\Stat::get('failed:' . $this->worker)); + } + + /** + * @expectedException \Resque\Exception + */ + public function testJobWithoutPerformMethodThrowsException() + { + $this->expectException(\Resque\Exception::class); + \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob_Without_Perform_Method'); + $job = $this->worker->reserve(); + $job->worker = $this->worker; + $job->perform(); + } + + /** + * @expectedException \Resque\Exception + */ + public function testInvalidJobThrowsException() + { + $this->expectException(\Resque\Exception::class); + \Resque\Resque::enqueue('jobs', '\Resque\Test\InvalidJob'); + $job = $this->worker->reserve(); + $job->worker = $this->worker; + $job->perform(); + } + + public function testJobWithSetUpCallbackFiresSetUp() + { + $payload = [ + 'class' => '\Resque\Test\TestJobWithSetUp', + 'args' => [ + 'somevar', + 'somevar2', + ], + ]; + $job = new \Resque\Job\Job('jobs', $payload); + $job->perform(); + + $this->assertTrue(TestJobWithSetUp::$called); + } + + public function testJobWithTearDownCallbackFiresTearDown() + { + $payload = [ + 'class' => '\Resque\Test\TestJobWithTearDown', + 'args' => [ + 'somevar', + 'somevar2', + ], + ]; + $job = new \Resque\Job\Job('jobs', $payload); + $job->perform(); + + $this->assertTrue(TestJobWithTearDown::$called); + } + + public function testNamespaceNaming() + { + $fixture = [ + ['test' => 'more:than:one:with:', 'assertValue' => 'more:than:one:with:'], + ['test' => 'more:than:one:without', 'assertValue' => 'more:than:one:without:'], + ['test' => 'resque', 'assertValue' => 'resque:'], + ['test' => 'resque:', 'assertValue' => 'resque:'], + ]; + + foreach ($fixture as $item) { + \Resque\Redis::prefix($item['test']); + $this->assertEquals(\Resque\Redis::getPrefix(), $item['assertValue']); + } + } + + public function testJobWithNamespace() + { + \Resque\Redis::prefix('php'); + $queue = 'jobs'; + $payload = ['another_value']; + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJobWithTearDown', $payload); + + $this->assertEquals(\Resque\Resque::queues(), ['jobs']); + $this->assertEquals(\Resque\Resque::size($queue), 1); + + \Resque\Redis::prefix('resque'); + $this->assertEquals(\Resque\Resque::size($queue), 0); + } + + public function testDequeueAll() + { + $queue = 'jobs'; + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJobDequeue'); + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJobDequeue'); + $this->assertEquals(\Resque\Resque::size($queue), 2); + $this->assertEquals(\Resque\Resque::dequeue($queue), 2); + $this->assertEquals(\Resque\Resque::size($queue), 0); + } + + public function testDequeueMakeSureNotDeleteOthers() + { + $queue = 'jobs'; + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJobDequeue'); + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJobDequeue'); + $other_queue = 'other_jobs'; + \Resque\Resque::enqueue($other_queue, '\Resque\Test\TestJobDequeue'); + \Resque\Resque::enqueue($other_queue, '\Resque\Test\TestJobDequeue'); + $this->assertEquals(\Resque\Resque::size($queue), 2); + $this->assertEquals(\Resque\Resque::size($other_queue), 2); + $this->assertEquals(\Resque\Resque::dequeue($queue), 2); + $this->assertEquals(\Resque\Resque::size($queue), 0); + $this->assertEquals(\Resque\Resque::size($other_queue), 2); + } + + public function testDequeueSpecificItem() + { + $queue = 'jobs'; + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJobDequeue1'); + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJobDequeue2'); + $this->assertEquals(\Resque\Resque::size($queue), 2); + $test = ['\Resque\Test\TestJobDequeue2']; + $this->assertEquals(\Resque\Resque::dequeue($queue, $test), 1); + $this->assertEquals(\Resque\Resque::size($queue), 1); + } + + public function testDequeueSpecificMultipleItems() + { + $queue = 'jobs'; + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJob_Dequeue1'); + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJob_Dequeue2'); + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJob_Dequeue3'); + $this->assertEquals(\Resque\Resque::size($queue), 3); + $test = ['\Resque\Test\TestJob_Dequeue2', '\Resque\Test\TestJob_Dequeue3']; + $this->assertEquals(\Resque\Resque::dequeue($queue, $test), 2); + $this->assertEquals(\Resque\Resque::size($queue), 1); + } + + public function testDequeueNonExistingItem() + { + $queue = 'jobs'; + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJob_Dequeue1'); + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJob_Dequeue2'); + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJob_Dequeue3'); + $this->assertEquals(\Resque\Resque::size($queue), 3); + $test = ['\Resque\Test\TestJob_Dequeue4']; + $this->assertEquals(\Resque\Resque::dequeue($queue, $test), 0); + $this->assertEquals(\Resque\Resque::size($queue), 3); + } + + public function testDequeueNonExistingItem2() + { + $queue = 'jobs'; + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJob_Dequeue1'); + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJob_Dequeue2'); + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJob_Dequeue3'); + $this->assertEquals(\Resque\Resque::size($queue), 3); + $test = ['\Resque\Test\TestJob_Dequeue4', '\Resque\Test\TestJob_Dequeue1']; + $this->assertEquals(\Resque\Resque::dequeue($queue, $test), 1); + $this->assertEquals(\Resque\Resque::size($queue), 2); + } + + public function testDequeueItemID() + { + $queue = 'jobs'; + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJob_Dequeue'); + $qid = \Resque\Resque::enqueue($queue, '\Resque\Test\TestJob_Dequeue'); + $this->assertEquals(\Resque\Resque::size($queue), 2); + $test = ['\Resque\Test\TestJob_Dequeue' => $qid]; + $this->assertEquals(\Resque\Resque::dequeue($queue, $test), 1); + $this->assertEquals(\Resque\Resque::size($queue), 1); + } + + public function testDequeueWrongItemID() + { + $queue = 'jobs'; + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJob_Dequeue'); + $qid = \Resque\Resque::enqueue($queue, '\Resque\Test\TestJob_Dequeue'); + $this->assertEquals(\Resque\Resque::size($queue), 2); + #qid right but class name is wrong + $test = ['\Resque\Test\TestJob_Dequeue1' => $qid]; + $this->assertEquals(\Resque\Resque::dequeue($queue, $test), 0); + $this->assertEquals(\Resque\Resque::size($queue), 2); + } + + public function testDequeueWrongItemID2() + { + $queue = 'jobs'; + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJob_Dequeue'); + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJob_Dequeue'); + $this->assertEquals(\Resque\Resque::size($queue), 2); + $test = ['\Resque\Test\TestJob_Dequeue' => 'r4nD0mH4sh3dId']; + $this->assertEquals(\Resque\Resque::dequeue($queue, $test), 0); + $this->assertEquals(\Resque\Resque::size($queue), 2); + } + + public function testDequeueItemWithArg() + { + $queue = 'jobs'; + $arg = ['foo' => 1, 'bar' => 2]; + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJobDequeue9'); + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJobDequeue9', $arg); + $this->assertEquals(\Resque\Resque::size($queue), 2); + $test = ['\Resque\Test\TestJobDequeue9' => $arg]; + $this->assertEquals(\Resque\Resque::dequeue($queue, $test), 1); + #$this->assertEquals(\Resque\Resque::size($queue), 1); + } + + public function testDequeueSeveralItemsWithArgs() + { + // GIVEN + $queue = 'jobs'; + $args = ['foo' => 1, 'bar' => 10]; + $removeArgs = ['foo' => 1, 'bar' => 2]; + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJobDequeue9', $args); + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJobDequeue9', $removeArgs); + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJobDequeue9', $removeArgs); + $this->assertEquals(\Resque\Resque::size($queue), 3, "Failed to add 3 items."); + + // WHEN + $test = ['\Resque\Test\TestJobDequeue9' => $removeArgs]; + $removedItems = \Resque\Resque::dequeue($queue, $test); + + // THEN + $this->assertEquals($removedItems, 2); + $this->assertEquals(\Resque\Resque::size($queue), 1); + $item = \Resque\Resque::pop($queue); + $this->assertIsArray($item['args']); + $this->assertEquals(10, $item['args'][0]['bar'], 'Wrong items were dequeued from queue!'); + } + + public function testDequeueItemWithUnorderedArg() + { + $queue = 'jobs'; + $arg = ['foo' => 1, 'bar' => 2]; + $arg2 = ['bar' => 2, 'foo' => 1]; + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJobDequeue'); + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJobDequeue', $arg); + $this->assertEquals(\Resque\Resque::size($queue), 2); + $test = ['\Resque\Test\TestJobDequeue' => $arg2]; + $this->assertEquals(\Resque\Resque::dequeue($queue, $test), 1); + $this->assertEquals(\Resque\Resque::size($queue), 1); + } + + public function testDequeueItemWithiWrongArg() + { + $queue = 'jobs'; + $arg = ['foo' => 1, 'bar' => 2]; + $arg2 = ['foo' => 2, 'bar' => 3]; + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJobDequeue'); + \Resque\Resque::enqueue($queue, '\Resque\Test\TestJobDequeue', $arg); + $this->assertEquals(\Resque\Resque::size($queue), 2); + $test = ['\Resque\Test\TestJobDequeue' => $arg2]; + $this->assertEquals(\Resque\Resque::dequeue($queue, $test), 0); + $this->assertEquals(\Resque\Resque::size($queue), 2); + } + + public function testUseDefaultFactoryToGetJobInstance() + { + $payload = [ + 'class' => '\Resque\Test\SomeJobClass', + 'args' => null + ]; + $job = new \Resque\Job\Job('jobs', $payload); + $instance = $job->getInstance(); + $this->assertInstanceOf('\Resque\Test\SomeJobClass', $instance); + } + + public function testUseFactoryToGetJobInstance() + { + $payload = [ + 'class' => 'SomeJobClass', + 'args' => [[]] + ]; + $job = new \Resque\Job\Job('jobs', $payload); + $factory = new Some_Stub_Factory(); + $job->setJobFactory($factory); + $instance = $job->getInstance(); + $this->assertInstanceOf('\Resque\Job\JobInterface', $instance); + } +} + +class SomeJobClass implements \Resque\Job\JobInterface +{ + public static $called = false; + public $args = false; + public $queue; + public $job; + + /** + * @return bool + */ + public function perform() + { + return true; + } +} + +class Some_Stub_Factory implements \Resque\Job\FactoryInterface +{ + public static $called = false; + public $args = false; + public $queue; + public $job; + + /** + * @param $className + * @param array $args + * @param $queue + * @return \Resque\Job\JobInterface + */ + public function create($className, $args, $queue) + { + return new SomeJobClass(); + } +} diff --git a/test/Resque/Tests/LogTest.php b/tests/Resque/Tests/LogTest.php similarity index 76% rename from test/Resque/Tests/LogTest.php rename to tests/Resque/Tests/LogTest.php index 76a74d1..18109b9 100644 --- a/test/Resque/Tests/LogTest.php +++ b/tests/Resque/Tests/LogTest.php @@ -1,18 +1,20 @@ + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -class Resque_Tests_LogTest extends Resque_Tests_TestCase +class LogTest extends TestCase { public function testLogInterpolate() { - $logger = new Resque_Log(); + $logger = new \Resque\Log(); $actual = $logger->interpolate('string {replace}', ['replace' => 'value']); $expected = 'string value'; @@ -21,7 +23,7 @@ class Resque_Tests_LogTest extends Resque_Tests_TestCase public function testLogInterpolateMutiple() { - $logger = new Resque_Log(); + $logger = new \Resque\Log(); $actual = $logger->interpolate( 'string {replace1} {replace2}', ['replace1' => 'value1', 'replace2' => 'value2'] diff --git a/test/Resque/Tests/RedisTest.php b/tests/Resque/Tests/RedisTest.php similarity index 87% rename from test/Resque/Tests/RedisTest.php rename to tests/Resque/Tests/RedisTest.php index 7e9a346..b834f93 100644 --- a/test/Resque/Tests/RedisTest.php +++ b/tests/Resque/Tests/RedisTest.php @@ -1,18 +1,25 @@ + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -class Resque_Tests_RedisTest extends Resque_Tests_TestCase +class RedisTest extends TestCase { public function testRedisGetSet() { - $this->redis->set("testKey", 24); + $this->redis->set( + 'testKey', + 24, + ['ex' => \Resque\Redis::DEFAULT_REDIS_TTL], + ); + $val = $this->redis->get("testKey"); $this->assertEquals(24, $val); } @@ -28,14 +35,14 @@ class Resque_Tests_RedisTest extends Resque_Tests_TestCase // Input , Expected output ['', [ 'localhost', - Resque_Redis::DEFAULT_PORT, + \Resque\Redis::DEFAULT_PORT, false, false, false, [], ]], ['localhost', [ 'localhost', - Resque_Redis::DEFAULT_PORT, + \Resque\Redis::DEFAULT_PORT, false, false, false, [], @@ -56,14 +63,14 @@ class Resque_Tests_RedisTest extends Resque_Tests_TestCase ]], ['redis://foobar', [ 'foobar', - Resque_Redis::DEFAULT_PORT, + \Resque\Redis::DEFAULT_PORT, false, false, false, [], ]], ['redis://foobar/', [ 'foobar', - Resque_Redis::DEFAULT_PORT, + \Resque\Redis::DEFAULT_PORT, false, false, false, [], @@ -175,18 +182,20 @@ class Resque_Tests_RedisTest extends Resque_Tests_TestCase */ public function testParsingValidDsnString($dsn, $expected) { - $result = Resque_Redis::parseDsn($dsn); + $result = \Resque\Redis::parseDsn($dsn); $this->assertEquals($expected, $result); } /** * @dataProvider bogusDsnStringProvider - * @expectedException InvalidArgumentException + * + * @expectedException \InvalidArgumentException + * * @param $dsn */ public function testParsingBogusDsnStringThrowsException($dsn) { - // The next line should throw an InvalidArgumentException - Resque_Redis::parseDsn($dsn); + $this->expectException(\InvalidArgumentException::class); + \Resque\Redis::parseDsn($dsn); } } \ No newline at end of file diff --git a/test/Resque/Tests/StatTest.php b/tests/Resque/Tests/StatTest.php similarity index 54% rename from test/Resque/Tests/StatTest.php rename to tests/Resque/Tests/StatTest.php index 121aaff..3b04c3d 100644 --- a/test/Resque/Tests/StatTest.php +++ b/tests/Resque/Tests/StatTest.php @@ -1,51 +1,53 @@ + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -class Resque_Tests_StatTest extends Resque_Tests_TestCase +class StatTest extends TestCase { public function testStatCanBeIncremented() { - Resque_Stat::incr('test_incr'); - Resque_Stat::incr('test_incr'); + \Resque\Stat::incr('test_incr'); + \Resque\Stat::incr('test_incr'); $this->assertEquals(2, $this->redis->get('resque:stat:test_incr')); } public function testStatCanBeIncrementedByX() { - Resque_Stat::incr('test_incrX', 10); - Resque_Stat::incr('test_incrX', 11); + \Resque\Stat::incr('test_incrX', 10); + \Resque\Stat::incr('test_incrX', 11); $this->assertEquals(21, $this->redis->get('resque:stat:test_incrX')); } public function testStatCanBeDecremented() { - Resque_Stat::incr('test_decr', 22); - Resque_Stat::decr('test_decr'); + \Resque\Stat::incr('test_decr', 22); + \Resque\Stat::decr('test_decr'); $this->assertEquals(21, $this->redis->get('resque:stat:test_decr')); } public function testStatCanBeDecrementedByX() { - Resque_Stat::incr('test_decrX', 22); - Resque_Stat::decr('test_decrX', 11); + \Resque\Stat::incr('test_decrX', 22); + \Resque\Stat::decr('test_decrX', 11); $this->assertEquals(11, $this->redis->get('resque:stat:test_decrX')); } public function testGetStatByName() { - Resque_Stat::incr('test_get', 100); - $this->assertEquals(100, Resque_Stat::get('test_get')); + \Resque\Stat::incr('test_get', 100); + $this->assertEquals(100, \Resque\Stat::get('test_get')); } public function testGetUnknownStatReturns0() { - $this->assertEquals(0, Resque_Stat::get('test_get_unknown')); + $this->assertEquals(0, \Resque\Stat::get('test_get_unknown')); } } \ No newline at end of file diff --git a/test/Resque/Tests/TestCase.php b/tests/Resque/Tests/TestCase.php similarity index 55% rename from test/Resque/Tests/TestCase.php rename to tests/Resque/Tests/TestCase.php index a428d12..d184010 100644 --- a/test/Resque/Tests/TestCase.php +++ b/tests/Resque/Tests/TestCase.php @@ -1,30 +1,31 @@ + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ - -class Resque_Tests_TestCase extends PHPUnit\Framework\TestCase +class TestCase extends \PHPUnit\Framework\TestCase { protected $resque; protected $redis; - public static function setUpBeforeClass() + public static function setUpBeforeClass(): void { date_default_timezone_set('UTC'); } - public function setUp() + public function setUp(): void { // Setup redis connection for testing. global $redisTestServer; - $this->redis = new Credis_Client($redisTestServer, '6379'); - Resque::setBackend($redisTestServer); + $this->redis = new \Credis_Client($redisTestServer, '6379'); + \Resque\Resque::setBackend($redisTestServer); $this->redis->flushAll(); } } diff --git a/test/Resque/Tests/WorkerTest.php b/tests/Resque/Tests/WorkerTest.php similarity index 55% rename from test/Resque/Tests/WorkerTest.php rename to tests/Resque/Tests/WorkerTest.php index 44f9c4c..e1b22e4 100644 --- a/test/Resque/Tests/WorkerTest.php +++ b/tests/Resque/Tests/WorkerTest.php @@ -1,19 +1,21 @@ + * @author Daniel Mason * @license http://www.opensource.org/licenses/mit-license.php */ -class Resque_Tests_WorkerTest extends Resque_Tests_TestCase +class WorkerTest extends TestCase { public function testWorkerRegistersInList() { - $worker = new Resque_Worker('*'); - $worker->setLogger(new Resque_Log()); + $worker = new \Resque\Worker('*'); + $worker->setLogger(new \Resque\Log()); $worker->registerWorker(); // Make sure the worker is in the list @@ -25,76 +27,76 @@ class Resque_Tests_WorkerTest extends Resque_Tests_TestCase $num = 3; // Register a few workers for ($i = 0; $i < $num; ++$i) { - $worker = new Resque_Worker('queue_' . $i); - $worker->setLogger(new Resque_Log()); + $worker = new \Resque\Worker('queue_' . $i); + $worker->setLogger(new \Resque\Log()); $worker->registerWorker(); } // Now try to get them - $this->assertEquals($num, count(Resque_Worker::all())); + $this->assertEquals($num, count(\Resque\Worker::all())); } public function testGetWorkerById() { - $worker = new Resque_Worker('*'); - $worker->setLogger(new Resque_Log()); + $worker = new \Resque\Worker('*'); + $worker->setLogger(new \Resque\Log()); $worker->registerWorker(); - $newWorker = Resque_Worker::find((string)$worker); + $newWorker = \Resque\Worker::find((string)$worker); $this->assertEquals((string)$worker, (string)$newWorker); } public function testInvalidWorkerDoesNotExist() { - $this->assertFalse(Resque_Worker::exists('blah')); + $this->assertFalse(\Resque\Worker::exists('blah')); } public function testWorkerCanUnregister() { - $worker = new Resque_Worker('*'); - $worker->setLogger(new Resque_Log()); + $worker = new \Resque\Worker('*'); + $worker->setLogger(new \Resque\Log()); $worker->registerWorker(); $worker->unregisterWorker(); - $this->assertFalse(Resque_Worker::exists((string)$worker)); - $this->assertEquals([], Resque_Worker::all()); + $this->assertFalse(\Resque\Worker::exists((string)$worker)); + $this->assertEquals([], \Resque\Worker::all()); $this->assertEquals([], $this->redis->smembers('resque:workers')); } public function testPausedWorkerDoesNotPickUpJobs() { - $worker = new Resque_Worker('*'); - $worker->setLogger(new Resque_Log()); + $worker = new \Resque\Worker('*'); + $worker->setLogger(new \Resque\Log()); $worker->pauseProcessing(); - Resque::enqueue('jobs', 'Test_Job'); + \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob'); $worker->work(0); $worker->work(0); - $this->assertEquals(0, Resque_Stat::get('processed')); + $this->assertEquals(0, \Resque\Stat::get('processed')); } public function testResumedWorkerPicksUpJobs() { - $worker = new Resque_Worker('*'); - $worker->setLogger(new Resque_Log()); + $worker = new \Resque\Worker('*'); + $worker->setLogger(new \Resque\Log()); $worker->pauseProcessing(); - Resque::enqueue('jobs', 'Test_Job'); + \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob'); $worker->work(0); - $this->assertEquals(0, Resque_Stat::get('processed')); + $this->assertEquals(0, \Resque\Stat::get('processed')); $worker->unPauseProcessing(); $worker->work(0); - $this->assertEquals(1, Resque_Stat::get('processed')); + $this->assertEquals(1, \Resque\Stat::get('processed')); } public function testWorkerCanWorkOverMultipleQueues() { - $worker = new Resque_Worker([ + $worker = new \Resque\Worker([ 'queue1', 'queue2' ]); - $worker->setLogger(new Resque_Log()); + $worker->setLogger(new \Resque\Log()); $worker->registerWorker(); - Resque::enqueue('queue1', 'Test_Job_1'); - Resque::enqueue('queue2', 'Test_Job_2'); + \Resque\Resque::enqueue('queue1', '\Resque\Test\TestJob_1'); + \Resque\Resque::enqueue('queue2', '\Resque\Test\TestJob_2'); $job = $worker->reserve(); $this->assertEquals('queue1', $job->queue); @@ -105,18 +107,18 @@ class Resque_Tests_WorkerTest extends Resque_Tests_TestCase public function testWorkerWorksQueuesInSpecifiedOrder() { - $worker = new Resque_Worker([ + $worker = new \Resque\Worker([ 'high', 'medium', 'low' ]); - $worker->setLogger(new Resque_Log()); + $worker->setLogger(new \Resque\Log()); $worker->registerWorker(); // Queue the jobs in a different order - Resque::enqueue('low', 'Test_Job_1'); - Resque::enqueue('high', 'Test_Job_2'); - Resque::enqueue('medium', 'Test_Job_3'); + \Resque\Resque::enqueue('low', '\Resque\Test\TestJob_1'); + \Resque\Resque::enqueue('high', '\Resque\Test\TestJob_2'); + \Resque\Resque::enqueue('medium', '\Resque\Test\TestJob_3'); // Now check we get the jobs back in the right order $job = $worker->reserve(); @@ -131,12 +133,12 @@ class Resque_Tests_WorkerTest extends Resque_Tests_TestCase public function testWildcardQueueWorkerWorksAllQueues() { - $worker = new Resque_Worker('*'); - $worker->setLogger(new Resque_Log()); + $worker = new \Resque\Worker('*'); + $worker->setLogger(new \Resque\Log()); $worker->registerWorker(); - Resque::enqueue('queue1', 'Test_Job_1'); - Resque::enqueue('queue2', 'Test_Job_2'); + \Resque\Resque::enqueue('queue1', '\Resque\Test\TestJob_1'); + \Resque\Resque::enqueue('queue2', '\Resque\Test\TestJob_2'); $job = $worker->reserve(); $this->assertEquals('queue1', $job->queue); @@ -147,19 +149,19 @@ class Resque_Tests_WorkerTest extends Resque_Tests_TestCase public function testWorkerDoesNotWorkOnUnknownQueues() { - $worker = new Resque_Worker('queue1'); - $worker->setLogger(new Resque_Log()); + $worker = new \Resque\Worker('queue1'); + $worker->setLogger(new \Resque\Log()); $worker->registerWorker(); - Resque::enqueue('queue2', 'Test_Job'); + \Resque\Resque::enqueue('queue2', '\Resque\Test\TestJob'); $this->assertFalse($worker->reserve()); } public function testWorkerClearsItsStatusWhenNotWorking() { - Resque::enqueue('jobs', 'Test_Job'); - $worker = new Resque_Worker('jobs'); - $worker->setLogger(new Resque_Log()); + \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob'); + $worker = new \Resque\Worker('jobs'); + $worker->setLogger(new \Resque\Log()); $job = $worker->reserve(); $worker->workingOn($job); $worker->doneWorking(); @@ -168,14 +170,14 @@ class Resque_Tests_WorkerTest extends Resque_Tests_TestCase public function testWorkerRecordsWhatItIsWorkingOn() { - $worker = new Resque_Worker('jobs'); - $worker->setLogger(new Resque_Log()); + $worker = new \Resque\Worker('jobs'); + $worker->setLogger(new \Resque\Log()); $worker->registerWorker(); $payload = [ - 'class' => 'Test_Job' + 'class' => '\Resque\Test\TestJob' ]; - $job = new Resque_Job('jobs', $payload); + $job = new \Resque\Job\Job('jobs', $payload); $worker->workingOn($job); $job = $worker->job(); @@ -188,14 +190,17 @@ class Resque_Tests_WorkerTest extends Resque_Tests_TestCase public function testWorkerErasesItsStatsWhenShutdown() { - Resque::enqueue('jobs', 'Test_Job'); - Resque::enqueue('jobs', 'Invalid_Job'); + \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob'); + \Resque\Resque::enqueue('jobs', '\Resque\Test\InvalidJob'); - $worker = new Resque_Worker('jobs'); - $worker->setLogger(new Resque_Log()); + $worker = new \Resque\Worker('jobs'); + $worker->setLogger(new \Resque\Log()); $worker->work(0); $worker->work(0); + // Allow time for async unlink to work + sleep(2); + $this->assertEquals(0, $worker->getStat('processed')); $this->assertEquals(0, $worker->getStat('failed')); } @@ -203,84 +208,84 @@ class Resque_Tests_WorkerTest extends Resque_Tests_TestCase public function testWorkerCleansUpDeadWorkersOnStartup() { // Register a good worker - $goodWorker = new Resque_Worker('jobs'); - $goodWorker->setLogger(new Resque_Log()); + $goodWorker = new \Resque\Worker('jobs'); + $goodWorker->setLogger(new \Resque\Log()); $goodWorker->registerWorker(); $workerId = explode(':', $goodWorker); // Register some bad workers - $worker = new Resque_Worker('jobs'); - $worker->setLogger(new Resque_Log()); + $worker = new \Resque\Worker('jobs'); + $worker->setLogger(new \Resque\Log()); $worker->setId($workerId[0] . ':1:jobs'); $worker->registerWorker(); - $worker = new Resque_Worker(['high', 'low']); - $worker->setLogger(new Resque_Log()); + $worker = new \Resque\Worker(['high', 'low']); + $worker->setLogger(new \Resque\Log()); $worker->setId($workerId[0] . ':2:high,low'); $worker->registerWorker(); - $this->assertEquals(3, count(Resque_Worker::all())); + $this->assertEquals(3, count(\Resque\Worker::all())); $goodWorker->pruneDeadWorkers(); // There should only be $goodWorker left now - $this->assertEquals(1, count(Resque_Worker::all())); + $this->assertEquals(1, count(\Resque\Worker::all())); } public function testDeadWorkerCleanUpDoesNotCleanUnknownWorkers() { // Register a bad worker on this machine - $worker = new Resque_Worker('jobs'); - $worker->setLogger(new Resque_Log()); + $worker = new \Resque\Worker('jobs'); + $worker->setLogger(new \Resque\Log()); $workerId = explode(':', $worker); $worker->setId($workerId[0] . ':1:jobs'); $worker->registerWorker(); // Register some other false workers - $worker = new Resque_Worker('jobs'); - $worker->setLogger(new Resque_Log()); + $worker = new \Resque\Worker('jobs'); + $worker->setLogger(new \Resque\Log()); $worker->setId('my.other.host:1:jobs'); $worker->registerWorker(); - $this->assertEquals(2, count(Resque_Worker::all())); + $this->assertEquals(2, count(\Resque\Worker::all())); $worker->pruneDeadWorkers(); // my.other.host should be left - $workers = Resque_Worker::all(); + $workers = \Resque\Worker::all(); $this->assertEquals(1, count($workers)); $this->assertEquals((string)$worker, (string)$workers[0]); } public function testWorkerFailsUncompletedJobsOnExit() { - $worker = new Resque_Worker('jobs'); - $worker->setLogger(new Resque_Log()); + $worker = new \Resque\Worker('jobs'); + $worker->setLogger(new \Resque\Log()); $worker->registerWorker(); $payload = [ - 'class' => 'Test_Job' + 'class' => '\Resque\Test\TestJob' ]; - $job = new Resque_Job('jobs', $payload); + $job = new \Resque\Job\Job('jobs', $payload); $worker->workingOn($job); $worker->unregisterWorker(); - $this->assertEquals(1, Resque_Stat::get('failed')); + $this->assertEquals(1, \Resque\Stat::get('failed')); } public function testBlockingListPop() { - $worker = new Resque_Worker('jobs'); - $worker->setLogger(new Resque_Log()); + $worker = new \Resque\Worker('jobs'); + $worker->setLogger(new \Resque\Log()); $worker->registerWorker(); - Resque::enqueue('jobs', 'Test_Job_1'); - Resque::enqueue('jobs', 'Test_Job_2'); + \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob_1'); + \Resque\Resque::enqueue('jobs', '\Resque\Test\TestJob_2'); $i = 1; while ($job = $worker->reserve(true, 2)) { - $this->assertEquals('Test_Job_' . $i, $job->payload['class']); + $this->assertEquals('\Resque\Test\TestJob_' . $i, $job->payload['class']); if ($i == 2) { break; diff --git a/test/bootstrap.php b/tests/bootstrap.php similarity index 57% rename from test/bootstrap.php rename to tests/bootstrap.php index 7488ff8..cf26e96 100644 --- a/test/bootstrap.php +++ b/tests/bootstrap.php @@ -1,4 +1,7 @@ add('Resque_Tests', __DIR__); # Redis configuration global $redisTestServer; $redisTestServer = getenv("REDIS_SERVER") ?: "redis"; -Resque::setBackend($redisTestServer); +\Resque\Resque::setBackend($redisTestServer); # Check Redis is accessable locally try { - $redisTest = new Resque_Redis($redisTestServer); -} catch (Exception $e) { - throw new Exception("Unable to connect to redis. Please check there is a redis-server running."); + $redisTest = new \Resque\Redis($redisTestServer); +} catch (\Exception $e) { + throw new \Exception("Unable to connect to redis. Please check there is a redis-server running."); } $redisTest = null; - - # Cleanup forked workers cleanly if (function_exists('pcntl_signal')) { - function sigint() - { - exit; - } - - pcntl_signal(SIGINT, 'sigint'); - pcntl_signal(SIGTERM, 'sigint'); + pcntl_signal(SIGINT, function() { exit; }); + pcntl_signal(SIGTERM, function() { exit; }); } # Bootstrap it -class Test_Job +class TestJob { public static $called = false; + public $args = false; + public $queue; + public $job; public function perform() { @@ -47,28 +45,33 @@ class Test_Job } } -class Failing_Job_Exception extends Exception +class FailingJobException extends \Exception { - } -class Failing_Job -{ - public function perform() - { - throw new Failing_Job_Exception('Message!'); - } -} - -class Test_Job_Without_Perform_Method -{ - -} - -class Test_Job_With_SetUp +class FailingJob { public static $called = false; public $args = false; + public $queue; + public $job; + + public function perform() + { + throw new FailingJobException('Message!'); + } +} + +class TestJobWithoutPerformMethod +{ +} + +class TestJobWithSetUp +{ + public static $called = false; + public $args = false; + public $queue; + public $job; public function setUp() { @@ -77,19 +80,19 @@ class Test_Job_With_SetUp public function perform() { - } } -class Test_Job_With_TearDown +class TestJobWithTearDown { public static $called = false; public $args = false; + public $queue; + public $job; public function perform() { - } public function tearDown()