diff --git a/.pubnub.yml b/.pubnub.yml index 2dc369e0..3f6c229d 100644 --- a/.pubnub.yml +++ b/.pubnub.yml @@ -1,8 +1,13 @@ name: php -version: 7.4.0 +version: 8.0.0 schema: 1 scm: github.com/pubnub/php changelog: + - date: 2025-03-19 + version: 8.0.0 + changes: + - type: improvement + text: "Replace dependency from Requests to GuzzleHTTP to allow communication over HTTP/2. This is potentially breaking change because removes the old way to set up custom transport with setting the client dependency. Read more in the documentation (migration guide available)." - date: 2025-02-18 version: 7.4.0 changes: @@ -452,8 +457,8 @@ sdks: - x86-64 - distribution-type: library distribution-repository: GitHub release - package-name: php-7.4.0.zip - location: https://github.com/pubnub/php/releases/tag/7.4.0 + package-name: php-8.0.0.zip + location: https://github.com/pubnub/php/releases/tag/8.0.0 requires: - name: rmccue/requests min-version: 1.0.0 diff --git a/CHANGELOG.md b/CHANGELOG.md index 999f04d1..6dc3d680 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,3 +1,9 @@ +## 8.0.0 +March 19 2025 + +#### Modified +- Replace dependency from Requests to GuzzleHTTP to allow communication over HTTP/2. This is potentially a breaking change because it removes the old way to set up custom transport with setting the client dependency. Read more in the documentation (migration guide available). + ## 7.4.0 February 18 2025 diff --git a/README.md b/README.md index 380f88b7..0cbeeeb8 100644 --- a/README.md +++ b/README.md @@ -41,7 +41,7 @@ You will need the publish and subscribe keys to authenticate your app. Get your { "require": { - "pubnub/pubnub": "7.4.0" + "pubnub/pubnub": "8.0.0" } } ``` diff --git a/composer.json b/composer.json index 53d1b7ec..1b509215 100644 --- a/composer.json +++ b/composer.json @@ -5,7 +5,7 @@ "keywords": ["api", "real-time", "realtime", "real time", "ajax", "push"], "homepage": "http://www.pubnub.com/", "license": "proprietary", - "version": "7.4.0", + "version": "8.0.0", "authors": [ { "name": "PubNub", @@ -24,13 +24,14 @@ ], "lint": [ "vendor/bin/phpstan analyze --memory-limit 256M", - "git diff --name-only --diff-filter=d origin/master HEAD | grep -E '\\.php$' | xargs vendor/bin/phpcs --standard=PSR12" + "git diff --name-only --diff-filter=d origin/master HEAD | grep -E '\\.php$' | xargs --no-run-if-empty vendor/bin/phpcs --standard=PSR12" ] }, "require": { - "php": ">=8.0", + "php": ">=8.1", "rmccue/requests": "^2.0", - "psr/log": "^1.1|^2.0|^3.0" + "psr/log": "^1.1|^2.0|^3.0", + "guzzlehttp/guzzle": "^7.9" }, "require-dev": { "monolog/monolog": "^2.9 || ^3.0", diff --git a/composer.lock b/composer.lock new file mode 100644 index 00000000..5b7342ae --- /dev/null +++ b/composer.lock @@ -0,0 +1,4480 @@ +{ + "_readme": [ + "This file locks the dependencies of your project to a known state", + "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", + "This file is @generated automatically" + ], + "content-hash": "049f9de47ba2d97bf73027fc71176620", + "packages": [ + { + "name": "guzzlehttp/guzzle", + "version": "7.9.2", + "source": { + "type": "git", + "url": "https://github.com/guzzle/guzzle.git", + "reference": "d281ed313b989f213357e3be1a179f02196ac99b" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/guzzle/guzzle/zipball/d281ed313b989f213357e3be1a179f02196ac99b", + "reference": "d281ed313b989f213357e3be1a179f02196ac99b", + "shasum": "" + }, + "require": { + "ext-json": "*", + "guzzlehttp/promises": "^1.5.3 || ^2.0.3", + "guzzlehttp/psr7": "^2.7.0", + "php": "^7.2.5 || ^8.0", + "psr/http-client": "^1.0", + "symfony/deprecation-contracts": "^2.2 || ^3.0" + }, + "provide": { + "psr/http-client-implementation": "1.0" + }, + "require-dev": { + "bamarni/composer-bin-plugin": "^1.8.2", + "ext-curl": "*", + "guzzle/client-integration-tests": "3.0.2", + "php-http/message-factory": "^1.1", + "phpunit/phpunit": "^8.5.39 || ^9.6.20", + "psr/log": "^1.1 || ^2.0 || ^3.0" + }, + "suggest": { + "ext-curl": "Required for CURL handler support", + "ext-intl": "Required for Internationalized Domain Name (IDN) support", + "psr/log": "Required for using the Log middleware" + }, + "type": "library", + "extra": { + "bamarni-bin": { + "bin-links": true, + "forward-command": false + } + }, + "autoload": { + "files": [ + "src/functions_include.php" + ], + "psr-4": { + "GuzzleHttp\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Graham Campbell", + "email": "hello@gjcampbell.co.uk", + "homepage": "https://github.com/GrahamCampbell" + }, + { + "name": "Michael Dowling", + "email": "mtdowling@gmail.com", + "homepage": "https://github.com/mtdowling" + }, + { + "name": "Jeremy Lindblom", + "email": "jeremeamia@gmail.com", + "homepage": "https://github.com/jeremeamia" + }, + { + "name": "George Mponos", + "email": "gmponos@gmail.com", + "homepage": "https://github.com/gmponos" + }, + { + "name": "Tobias Nyholm", + "email": "tobias.nyholm@gmail.com", + "homepage": "https://github.com/Nyholm" + }, + { + "name": "Márk Sági-Kazár", + "email": "mark.sagikazar@gmail.com", + "homepage": "https://github.com/sagikazarmark" + }, + { + "name": "Tobias Schultze", + "email": "webmaster@tubo-world.de", + "homepage": "https://github.com/Tobion" + } + ], + "description": "Guzzle is a PHP HTTP client library", + "keywords": [ + "client", + "curl", + "framework", + "http", + "http client", + "psr-18", + "psr-7", + "rest", + "web service" + ], + "support": { + "issues": "https://github.com/guzzle/guzzle/issues", + "source": "https://github.com/guzzle/guzzle/tree/7.9.2" + }, + "funding": [ + { + "url": "https://github.com/GrahamCampbell", + "type": "github" + }, + { + "url": "https://github.com/Nyholm", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/guzzlehttp/guzzle", + "type": "tidelift" + } + ], + "time": "2024-07-24T11:22:20+00:00" + }, + { + "name": "guzzlehttp/promises", + "version": "2.0.4", + "source": { + "type": "git", + "url": "https://github.com/guzzle/promises.git", + "reference": "f9c436286ab2892c7db7be8c8da4ef61ccf7b455" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/guzzle/promises/zipball/f9c436286ab2892c7db7be8c8da4ef61ccf7b455", + "reference": "f9c436286ab2892c7db7be8c8da4ef61ccf7b455", + "shasum": "" + }, + "require": { + "php": "^7.2.5 || ^8.0" + }, + "require-dev": { + "bamarni/composer-bin-plugin": "^1.8.2", + "phpunit/phpunit": "^8.5.39 || ^9.6.20" + }, + "type": "library", + "extra": { + "bamarni-bin": { + "bin-links": true, + "forward-command": false + } + }, + "autoload": { + "psr-4": { + "GuzzleHttp\\Promise\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Graham Campbell", + "email": "hello@gjcampbell.co.uk", + "homepage": "https://github.com/GrahamCampbell" + }, + { + "name": "Michael Dowling", + "email": "mtdowling@gmail.com", + "homepage": "https://github.com/mtdowling" + }, + { + "name": "Tobias Nyholm", + "email": "tobias.nyholm@gmail.com", + "homepage": "https://github.com/Nyholm" + }, + { + "name": "Tobias Schultze", + "email": "webmaster@tubo-world.de", + "homepage": "https://github.com/Tobion" + } + ], + "description": "Guzzle promises library", + "keywords": [ + "promise" + ], + "support": { + "issues": "https://github.com/guzzle/promises/issues", + "source": "https://github.com/guzzle/promises/tree/2.0.4" + }, + "funding": [ + { + "url": "https://github.com/GrahamCampbell", + "type": "github" + }, + { + "url": "https://github.com/Nyholm", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/guzzlehttp/promises", + "type": "tidelift" + } + ], + "time": "2024-10-17T10:06:22+00:00" + }, + { + "name": "guzzlehttp/psr7", + "version": "2.7.0", + "source": { + "type": "git", + "url": "https://github.com/guzzle/psr7.git", + "reference": "a70f5c95fb43bc83f07c9c948baa0dc1829bf201" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/guzzle/psr7/zipball/a70f5c95fb43bc83f07c9c948baa0dc1829bf201", + "reference": "a70f5c95fb43bc83f07c9c948baa0dc1829bf201", + "shasum": "" + }, + "require": { + "php": "^7.2.5 || ^8.0", + "psr/http-factory": "^1.0", + "psr/http-message": "^1.1 || ^2.0", + "ralouphie/getallheaders": "^3.0" + }, + "provide": { + "psr/http-factory-implementation": "1.0", + "psr/http-message-implementation": "1.0" + }, + "require-dev": { + "bamarni/composer-bin-plugin": "^1.8.2", + "http-interop/http-factory-tests": "0.9.0", + "phpunit/phpunit": "^8.5.39 || ^9.6.20" + }, + "suggest": { + "laminas/laminas-httphandlerrunner": "Emit PSR-7 responses" + }, + "type": "library", + "extra": { + "bamarni-bin": { + "bin-links": true, + "forward-command": false + } + }, + "autoload": { + "psr-4": { + "GuzzleHttp\\Psr7\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Graham Campbell", + "email": "hello@gjcampbell.co.uk", + "homepage": "https://github.com/GrahamCampbell" + }, + { + "name": "Michael Dowling", + "email": "mtdowling@gmail.com", + "homepage": "https://github.com/mtdowling" + }, + { + "name": "George Mponos", + "email": "gmponos@gmail.com", + "homepage": "https://github.com/gmponos" + }, + { + "name": "Tobias Nyholm", + "email": "tobias.nyholm@gmail.com", + "homepage": "https://github.com/Nyholm" + }, + { + "name": "Márk Sági-Kazár", + "email": "mark.sagikazar@gmail.com", + "homepage": "https://github.com/sagikazarmark" + }, + { + "name": "Tobias Schultze", + "email": "webmaster@tubo-world.de", + "homepage": "https://github.com/Tobion" + }, + { + "name": "Márk Sági-Kazár", + "email": "mark.sagikazar@gmail.com", + "homepage": "https://sagikazarmark.hu" + } + ], + "description": "PSR-7 message implementation that also provides common utility methods", + "keywords": [ + "http", + "message", + "psr-7", + "request", + "response", + "stream", + "uri", + "url" + ], + "support": { + "issues": "https://github.com/guzzle/psr7/issues", + "source": "https://github.com/guzzle/psr7/tree/2.7.0" + }, + "funding": [ + { + "url": "https://github.com/GrahamCampbell", + "type": "github" + }, + { + "url": "https://github.com/Nyholm", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/guzzlehttp/psr7", + "type": "tidelift" + } + ], + "time": "2024-07-18T11:15:46+00:00" + }, + { + "name": "psr/http-client", + "version": "1.0.3", + "source": { + "type": "git", + "url": "https://github.com/php-fig/http-client.git", + "reference": "bb5906edc1c324c9a05aa0873d40117941e5fa90" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/php-fig/http-client/zipball/bb5906edc1c324c9a05aa0873d40117941e5fa90", + "reference": "bb5906edc1c324c9a05aa0873d40117941e5fa90", + "shasum": "" + }, + "require": { + "php": "^7.0 || ^8.0", + "psr/http-message": "^1.0 || ^2.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.0.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Http\\Client\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "https://www.php-fig.org/" + } + ], + "description": "Common interface for HTTP clients", + "homepage": "https://github.com/php-fig/http-client", + "keywords": [ + "http", + "http-client", + "psr", + "psr-18" + ], + "support": { + "source": "https://github.com/php-fig/http-client" + }, + "time": "2023-09-23T14:17:50+00:00" + }, + { + "name": "psr/http-factory", + "version": "1.1.0", + "source": { + "type": "git", + "url": "https://github.com/php-fig/http-factory.git", + "reference": "2b4765fddfe3b508ac62f829e852b1501d3f6e8a" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/php-fig/http-factory/zipball/2b4765fddfe3b508ac62f829e852b1501d3f6e8a", + "reference": "2b4765fddfe3b508ac62f829e852b1501d3f6e8a", + "shasum": "" + }, + "require": { + "php": ">=7.1", + "psr/http-message": "^1.0 || ^2.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.0.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Http\\Message\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "https://www.php-fig.org/" + } + ], + "description": "PSR-17: Common interfaces for PSR-7 HTTP message factories", + "keywords": [ + "factory", + "http", + "message", + "psr", + "psr-17", + "psr-7", + "request", + "response" + ], + "support": { + "source": "https://github.com/php-fig/http-factory" + }, + "time": "2024-04-15T12:06:14+00:00" + }, + { + "name": "psr/http-message", + "version": "2.0", + "source": { + "type": "git", + "url": "https://github.com/php-fig/http-message.git", + "reference": "402d35bcb92c70c026d1a6a9883f06b2ead23d71" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/php-fig/http-message/zipball/402d35bcb92c70c026d1a6a9883f06b2ead23d71", + "reference": "402d35bcb92c70c026d1a6a9883f06b2ead23d71", + "shasum": "" + }, + "require": { + "php": "^7.2 || ^8.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "2.0.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Http\\Message\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "https://www.php-fig.org/" + } + ], + "description": "Common interface for HTTP messages", + "homepage": "https://github.com/php-fig/http-message", + "keywords": [ + "http", + "http-message", + "psr", + "psr-7", + "request", + "response" + ], + "support": { + "source": "https://github.com/php-fig/http-message/tree/2.0" + }, + "time": "2023-04-04T09:54:51+00:00" + }, + { + "name": "psr/log", + "version": "3.0.2", + "source": { + "type": "git", + "url": "https://github.com/php-fig/log.git", + "reference": "f16e1d5863e37f8d8c2a01719f5b34baa2b714d3" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/php-fig/log/zipball/f16e1d5863e37f8d8c2a01719f5b34baa2b714d3", + "reference": "f16e1d5863e37f8d8c2a01719f5b34baa2b714d3", + "shasum": "" + }, + "require": { + "php": ">=8.0.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "3.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Log\\": "src" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "https://www.php-fig.org/" + } + ], + "description": "Common interface for logging libraries", + "homepage": "https://github.com/php-fig/log", + "keywords": [ + "log", + "psr", + "psr-3" + ], + "support": { + "source": "https://github.com/php-fig/log/tree/3.0.2" + }, + "time": "2024-09-11T13:17:53+00:00" + }, + { + "name": "ralouphie/getallheaders", + "version": "3.0.3", + "source": { + "type": "git", + "url": "https://github.com/ralouphie/getallheaders.git", + "reference": "120b605dfeb996808c31b6477290a714d356e822" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/ralouphie/getallheaders/zipball/120b605dfeb996808c31b6477290a714d356e822", + "reference": "120b605dfeb996808c31b6477290a714d356e822", + "shasum": "" + }, + "require": { + "php": ">=5.6" + }, + "require-dev": { + "php-coveralls/php-coveralls": "^2.1", + "phpunit/phpunit": "^5 || ^6.5" + }, + "type": "library", + "autoload": { + "files": [ + "src/getallheaders.php" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Ralph Khattar", + "email": "ralph.khattar@gmail.com" + } + ], + "description": "A polyfill for getallheaders.", + "support": { + "issues": "https://github.com/ralouphie/getallheaders/issues", + "source": "https://github.com/ralouphie/getallheaders/tree/develop" + }, + "time": "2019-03-08T08:55:37+00:00" + }, + { + "name": "rmccue/requests", + "version": "v2.0.15", + "source": { + "type": "git", + "url": "https://github.com/WordPress/Requests.git", + "reference": "877cd66169755899682f1595e057334b40d9d149" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/WordPress/Requests/zipball/877cd66169755899682f1595e057334b40d9d149", + "reference": "877cd66169755899682f1595e057334b40d9d149", + "shasum": "" + }, + "require": { + "ext-json": "*", + "php": ">=5.6" + }, + "require-dev": { + "dealerdirect/phpcodesniffer-composer-installer": "^0.7", + "php-parallel-lint/php-console-highlighter": "^0.5.0", + "php-parallel-lint/php-parallel-lint": "^1.3.1", + "phpcompatibility/php-compatibility": "^9.0", + "requests/test-server": "dev-main", + "roave/security-advisories": "dev-latest", + "squizlabs/php_codesniffer": "^3.6", + "wp-coding-standards/wpcs": "^2.0", + "yoast/phpunit-polyfills": "^1.0.0" + }, + "suggest": { + "art4/requests-psr18-adapter": "For using Requests as a PSR-18 HTTP Client", + "ext-curl": "For improved performance", + "ext-openssl": "For secure transport support", + "ext-zlib": "For improved performance when decompressing encoded streams" + }, + "type": "library", + "autoload": { + "files": [ + "library/Deprecated.php" + ], + "psr-4": { + "WpOrg\\Requests\\": "src/" + }, + "classmap": [ + "library/Requests.php" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "ISC" + ], + "authors": [ + { + "name": "Ryan McCue", + "homepage": "https://rmccue.io/" + }, + { + "name": "Alain Schlesser", + "homepage": "https://github.com/schlessera" + }, + { + "name": "Juliette Reinders Folmer", + "homepage": "https://github.com/jrfnl" + }, + { + "name": "Contributors", + "homepage": "https://github.com/WordPress/Requests/graphs/contributors" + } + ], + "description": "A HTTP library written in PHP, for human beings.", + "homepage": "https://requests.ryanmccue.info/", + "keywords": [ + "curl", + "fsockopen", + "http", + "idna", + "ipv6", + "iri", + "sockets" + ], + "support": { + "docs": "https://requests.ryanmccue.info/", + "issues": "https://github.com/WordPress/Requests/issues", + "source": "https://github.com/WordPress/Requests" + }, + "time": "2025-01-21T10:13:31+00:00" + }, + { + "name": "symfony/deprecation-contracts", + "version": "v3.5.1", + "source": { + "type": "git", + "url": "https://github.com/symfony/deprecation-contracts.git", + "reference": "74c71c939a79f7d5bf3c1ce9f5ea37ba0114c6f6" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/deprecation-contracts/zipball/74c71c939a79f7d5bf3c1ce9f5ea37ba0114c6f6", + "reference": "74c71c939a79f7d5bf3c1ce9f5ea37ba0114c6f6", + "shasum": "" + }, + "require": { + "php": ">=8.1" + }, + "type": "library", + "extra": { + "thanks": { + "url": "https://github.com/symfony/contracts", + "name": "symfony/contracts" + }, + "branch-alias": { + "dev-main": "3.5-dev" + } + }, + "autoload": { + "files": [ + "function.php" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "A generic function and convention to trigger deprecation notices", + "homepage": "https://symfony.com", + "support": { + "source": "https://github.com/symfony/deprecation-contracts/tree/v3.5.1" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2024-09-25T14:20:29+00:00" + } + ], + "packages-dev": [ + { + "name": "behat/behat", + "version": "v3.19.0", + "source": { + "type": "git", + "url": "https://github.com/Behat/Behat.git", + "reference": "6cf82375a88145e33e10a34b211a3f914cbd02ee" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/Behat/Behat/zipball/6cf82375a88145e33e10a34b211a3f914cbd02ee", + "reference": "6cf82375a88145e33e10a34b211a3f914cbd02ee", + "shasum": "" + }, + "require": { + "behat/gherkin": "^4.10.0", + "behat/transliterator": "^1.5", + "composer-runtime-api": "^2.2", + "composer/xdebug-handler": "^3.0", + "ext-mbstring": "*", + "php": "8.1.* || 8.2.* || 8.3.* || 8.4.* ", + "psr/container": "^1.0 || ^2.0", + "symfony/config": "^5.4 || ^6.4 || ^7.0", + "symfony/console": "^5.4 || ^6.4 || ^7.0", + "symfony/dependency-injection": "^5.4 || ^6.4 || ^7.0", + "symfony/event-dispatcher": "^5.4 || ^6.4 || ^7.0", + "symfony/translation": "^5.4 || ^6.4 || ^7.0", + "symfony/yaml": "^5.4 || ^6.4 || ^7.0" + }, + "require-dev": { + "friendsofphp/php-cs-fixer": "^3.68", + "phpstan/phpstan": "^2.0", + "phpunit/phpunit": "^9.6", + "sebastian/diff": "^4.0", + "symfony/polyfill-php84": "^1.31", + "symfony/process": "^5.4 || ^6.4 || ^7.0" + }, + "suggest": { + "ext-dom": "Needed to output test results in JUnit format." + }, + "bin": [ + "bin/behat" + ], + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "3.x-dev" + } + }, + "autoload": { + "psr-4": { + "Behat\\Hook\\": "src/Behat/Hook/", + "Behat\\Step\\": "src/Behat/Step/", + "Behat\\Behat\\": "src/Behat/Behat/", + "Behat\\Config\\": "src/Behat/Config/", + "Behat\\Testwork\\": "src/Behat/Testwork/", + "Behat\\Transformation\\": "src/Behat/Transformation/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Konstantin Kudryashov", + "email": "ever.zet@gmail.com", + "homepage": "http://everzet.com" + } + ], + "description": "Scenario-oriented BDD framework for PHP", + "homepage": "https://behat.org/", + "keywords": [ + "Agile", + "BDD", + "ScenarioBDD", + "Scrum", + "StoryBDD", + "User story", + "business", + "development", + "documentation", + "examples", + "symfony", + "testing" + ], + "support": { + "issues": "https://github.com/Behat/Behat/issues", + "source": "https://github.com/Behat/Behat/tree/v3.19.0" + }, + "time": "2025-02-13T09:07:11+00:00" + }, + { + "name": "behat/gherkin", + "version": "v4.12.0", + "source": { + "type": "git", + "url": "https://github.com/Behat/Gherkin.git", + "reference": "cc3a7e224b36373be382b53ef02ede0f1807bb58" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/Behat/Gherkin/zipball/cc3a7e224b36373be382b53ef02ede0f1807bb58", + "reference": "cc3a7e224b36373be382b53ef02ede0f1807bb58", + "shasum": "" + }, + "require": { + "composer-runtime-api": "^2.2", + "php": "8.1.* || 8.2.* || 8.3.* || 8.4.*" + }, + "require-dev": { + "cucumber/cucumber": "dev-gherkin-24.1.0", + "friendsofphp/php-cs-fixer": "^3.65", + "phpstan/extension-installer": "^1", + "phpstan/phpstan": "^2", + "phpstan/phpstan-phpunit": "^2", + "phpunit/phpunit": "^10.5", + "symfony/filesystem": "^5.4 || ^6.4 || ^7.0", + "symfony/yaml": "^5.4 || ^6.4 || ^7.0" + }, + "suggest": { + "symfony/yaml": "If you want to parse features, represented in YAML files" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "4.x-dev" + } + }, + "autoload": { + "psr-0": { + "Behat\\Gherkin": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Konstantin Kudryashov", + "email": "ever.zet@gmail.com", + "homepage": "https://everzet.com" + } + ], + "description": "Gherkin DSL parser for PHP", + "homepage": "https://behat.org/", + "keywords": [ + "BDD", + "Behat", + "Cucumber", + "DSL", + "gherkin", + "parser" + ], + "support": { + "issues": "https://github.com/Behat/Gherkin/issues", + "source": "https://github.com/Behat/Gherkin/tree/v4.12.0" + }, + "time": "2025-02-26T14:28:23+00:00" + }, + { + "name": "behat/transliterator", + "version": "v1.5.0", + "source": { + "type": "git", + "url": "https://github.com/Behat/Transliterator.git", + "reference": "baac5873bac3749887d28ab68e2f74db3a4408af" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/Behat/Transliterator/zipball/baac5873bac3749887d28ab68e2f74db3a4408af", + "reference": "baac5873bac3749887d28ab68e2f74db3a4408af", + "shasum": "" + }, + "require": { + "php": ">=7.2" + }, + "require-dev": { + "chuyskywalker/rolling-curl": "^3.1", + "php-yaoi/php-yaoi": "^1.0", + "phpunit/phpunit": "^8.5.25 || ^9.5.19" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.x-dev" + } + }, + "autoload": { + "psr-4": { + "Behat\\Transliterator\\": "src/Behat/Transliterator" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "Artistic-1.0" + ], + "description": "String transliterator", + "keywords": [ + "i18n", + "slug", + "transliterator" + ], + "support": { + "issues": "https://github.com/Behat/Transliterator/issues", + "source": "https://github.com/Behat/Transliterator/tree/v1.5.0" + }, + "time": "2022-03-30T09:27:43+00:00" + }, + { + "name": "composer/pcre", + "version": "3.3.2", + "source": { + "type": "git", + "url": "https://github.com/composer/pcre.git", + "reference": "b2bed4734f0cc156ee1fe9c0da2550420d99a21e" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/composer/pcre/zipball/b2bed4734f0cc156ee1fe9c0da2550420d99a21e", + "reference": "b2bed4734f0cc156ee1fe9c0da2550420d99a21e", + "shasum": "" + }, + "require": { + "php": "^7.4 || ^8.0" + }, + "conflict": { + "phpstan/phpstan": "<1.11.10" + }, + "require-dev": { + "phpstan/phpstan": "^1.12 || ^2", + "phpstan/phpstan-strict-rules": "^1 || ^2", + "phpunit/phpunit": "^8 || ^9" + }, + "type": "library", + "extra": { + "phpstan": { + "includes": [ + "extension.neon" + ] + }, + "branch-alias": { + "dev-main": "3.x-dev" + } + }, + "autoload": { + "psr-4": { + "Composer\\Pcre\\": "src" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Jordi Boggiano", + "email": "j.boggiano@seld.be", + "homepage": "http://seld.be" + } + ], + "description": "PCRE wrapping library that offers type-safe preg_* replacements.", + "keywords": [ + "PCRE", + "preg", + "regex", + "regular expression" + ], + "support": { + "issues": "https://github.com/composer/pcre/issues", + "source": "https://github.com/composer/pcre/tree/3.3.2" + }, + "funding": [ + { + "url": "https://packagist.com", + "type": "custom" + }, + { + "url": "https://github.com/composer", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/composer/composer", + "type": "tidelift" + } + ], + "time": "2024-11-12T16:29:46+00:00" + }, + { + "name": "composer/xdebug-handler", + "version": "3.0.5", + "source": { + "type": "git", + "url": "https://github.com/composer/xdebug-handler.git", + "reference": "6c1925561632e83d60a44492e0b344cf48ab85ef" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/composer/xdebug-handler/zipball/6c1925561632e83d60a44492e0b344cf48ab85ef", + "reference": "6c1925561632e83d60a44492e0b344cf48ab85ef", + "shasum": "" + }, + "require": { + "composer/pcre": "^1 || ^2 || ^3", + "php": "^7.2.5 || ^8.0", + "psr/log": "^1 || ^2 || ^3" + }, + "require-dev": { + "phpstan/phpstan": "^1.0", + "phpstan/phpstan-strict-rules": "^1.1", + "phpunit/phpunit": "^8.5 || ^9.6 || ^10.5" + }, + "type": "library", + "autoload": { + "psr-4": { + "Composer\\XdebugHandler\\": "src" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "John Stevenson", + "email": "john-stevenson@blueyonder.co.uk" + } + ], + "description": "Restarts a process without Xdebug.", + "keywords": [ + "Xdebug", + "performance" + ], + "support": { + "irc": "ircs://irc.libera.chat:6697/composer", + "issues": "https://github.com/composer/xdebug-handler/issues", + "source": "https://github.com/composer/xdebug-handler/tree/3.0.5" + }, + "funding": [ + { + "url": "https://packagist.com", + "type": "custom" + }, + { + "url": "https://github.com/composer", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/composer/composer", + "type": "tidelift" + } + ], + "time": "2024-05-06T16:37:16+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/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Marco Pivetta", + "email": "ocramius@gmail.com", + "homepage": "https://ocramius.github.io/" + } + ], + "description": "A small, lightweight utility to instantiate objects in PHP without invoking their constructors", + "homepage": "https://www.doctrine-project.org/projects/instantiator.html", + "keywords": [ + "constructor", + "instantiate" + ], + "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": "monolog/monolog", + "version": "3.8.1", + "source": { + "type": "git", + "url": "https://github.com/Seldaek/monolog.git", + "reference": "aef6ee73a77a66e404dd6540934a9ef1b3c855b4" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/Seldaek/monolog/zipball/aef6ee73a77a66e404dd6540934a9ef1b3c855b4", + "reference": "aef6ee73a77a66e404dd6540934a9ef1b3c855b4", + "shasum": "" + }, + "require": { + "php": ">=8.1", + "psr/log": "^2.0 || ^3.0" + }, + "provide": { + "psr/log-implementation": "3.0.0" + }, + "require-dev": { + "aws/aws-sdk-php": "^3.0", + "doctrine/couchdb": "~1.0@dev", + "elasticsearch/elasticsearch": "^7 || ^8", + "ext-json": "*", + "graylog2/gelf-php": "^1.4.2 || ^2.0", + "guzzlehttp/guzzle": "^7.4.5", + "guzzlehttp/psr7": "^2.2", + "mongodb/mongodb": "^1.8", + "php-amqplib/php-amqplib": "~2.4 || ^3", + "php-console/php-console": "^3.1.8", + "phpstan/phpstan": "^2", + "phpstan/phpstan-deprecation-rules": "^2", + "phpstan/phpstan-strict-rules": "^2", + "phpunit/phpunit": "^10.5.17 || ^11.0.7", + "predis/predis": "^1.1 || ^2", + "rollbar/rollbar": "^4.0", + "ruflin/elastica": "^7 || ^8", + "symfony/mailer": "^5.4 || ^6", + "symfony/mime": "^5.4 || ^6" + }, + "suggest": { + "aws/aws-sdk-php": "Allow sending log messages to AWS services like DynamoDB", + "doctrine/couchdb": "Allow sending log messages to a CouchDB server", + "elasticsearch/elasticsearch": "Allow sending log messages to an Elasticsearch server via official client", + "ext-amqp": "Allow sending log messages to an AMQP server (1.0+ required)", + "ext-curl": "Required to send log messages using the IFTTTHandler, the LogglyHandler, the SendGridHandler, the SlackWebhookHandler or the TelegramBotHandler", + "ext-mbstring": "Allow to work properly with unicode symbols", + "ext-mongodb": "Allow sending log messages to a MongoDB server (via driver)", + "ext-openssl": "Required to send log messages using SSL", + "ext-sockets": "Allow sending log messages to a Syslog server (via UDP driver)", + "graylog2/gelf-php": "Allow sending log messages to a GrayLog2 server", + "mongodb/mongodb": "Allow sending log messages to a MongoDB server (via library)", + "php-amqplib/php-amqplib": "Allow sending log messages to an AMQP server using php-amqplib", + "rollbar/rollbar": "Allow sending log messages to Rollbar", + "ruflin/elastica": "Allow sending log messages to an Elastic Search server" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "3.x-dev" + } + }, + "autoload": { + "psr-4": { + "Monolog\\": "src/Monolog" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Jordi Boggiano", + "email": "j.boggiano@seld.be", + "homepage": "https://seld.be" + } + ], + "description": "Sends your logs to files, sockets, inboxes, databases and various web services", + "homepage": "https://github.com/Seldaek/monolog", + "keywords": [ + "log", + "logging", + "psr-3" + ], + "support": { + "issues": "https://github.com/Seldaek/monolog/issues", + "source": "https://github.com/Seldaek/monolog/tree/3.8.1" + }, + "funding": [ + { + "url": "https://github.com/Seldaek", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/monolog/monolog", + "type": "tidelift" + } + ], + "time": "2024-12-05T17:15:07+00:00" + }, + { + "name": "myclabs/deep-copy", + "version": "1.13.0", + "source": { + "type": "git", + "url": "https://github.com/myclabs/DeepCopy.git", + "reference": "024473a478be9df5fdaca2c793f2232fe788e414" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/myclabs/DeepCopy/zipball/024473a478be9df5fdaca2c793f2232fe788e414", + "reference": "024473a478be9df5fdaca2c793f2232fe788e414", + "shasum": "" + }, + "require": { + "php": "^7.1 || ^8.0" + }, + "conflict": { + "doctrine/collections": "<1.6.8", + "doctrine/common": "<2.13.3 || >=3 <3.2.2" + }, + "require-dev": { + "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": { + "files": [ + "src/DeepCopy/deep_copy.php" + ], + "psr-4": { + "DeepCopy\\": "src/DeepCopy/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "description": "Create deep copies (clones) of your objects", + "keywords": [ + "clone", + "copy", + "duplicate", + "object", + "object graph" + ], + "support": { + "issues": "https://github.com/myclabs/DeepCopy/issues", + "source": "https://github.com/myclabs/DeepCopy/tree/1.13.0" + }, + "funding": [ + { + "url": "https://tidelift.com/funding/github/packagist/myclabs/deep-copy", + "type": "tidelift" + } + ], + "time": "2025-02-12T12:17:51+00:00" + }, + { + "name": "nikic/php-parser", + "version": "v5.4.0", + "source": { + "type": "git", + "url": "https://github.com/nikic/PHP-Parser.git", + "reference": "447a020a1f875a434d62f2a401f53b82a396e494" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/nikic/PHP-Parser/zipball/447a020a1f875a434d62f2a401f53b82a396e494", + "reference": "447a020a1f875a434d62f2a401f53b82a396e494", + "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.4.0" + }, + "time": "2024-12-30T11:07:19+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": "*", + "ext-xmlwriter": "*", + "phar-io/version": "^3.0.1", + "php": "^7.2 || ^8.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "2.0.x-dev" + } + }, + "autoload": { + "classmap": [ + "src/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Arne Blankerts", + "email": "arne@blankerts.de", + "role": "Developer" + }, + { + "name": "Sebastian Heuer", + "email": "sebastian@phpeople.de", + "role": "Developer" + }, + { + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de", + "role": "Developer" + } + ], + "description": "Component for reading phar.io manifest information from a PHP Archive (PHAR)", + "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": "3.2.1", + "source": { + "type": "git", + "url": "https://github.com/phar-io/version.git", + "reference": "4f7fd7836c6f332bb2933569e566a0d6c4cbed74" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/phar-io/version/zipball/4f7fd7836c6f332bb2933569e566a0d6c4cbed74", + "reference": "4f7fd7836c6f332bb2933569e566a0d6c4cbed74", + "shasum": "" + }, + "require": { + "php": "^7.2 || ^8.0" + }, + "type": "library", + "autoload": { + "classmap": [ + "src/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Arne Blankerts", + "email": "arne@blankerts.de", + "role": "Developer" + }, + { + "name": "Sebastian Heuer", + "email": "sebastian@phpeople.de", + "role": "Developer" + }, + { + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de", + "role": "Developer" + } + ], + "description": "Library for handling version information and constraints", + "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": "phpstan/phpstan", + "version": "1.12.21", + "source": { + "type": "git", + "url": "https://github.com/phpstan/phpstan.git", + "reference": "14276fdef70575106a3392a4ed553c06a984df28" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/phpstan/phpstan/zipball/14276fdef70575106a3392a4ed553c06a984df28", + "reference": "14276fdef70575106a3392a4ed553c06a984df28", + "shasum": "" + }, + "require": { + "php": "^7.2|^8.0" + }, + "conflict": { + "phpstan/phpstan-shim": "*" + }, + "bin": [ + "phpstan", + "phpstan.phar" + ], + "type": "library", + "autoload": { + "files": [ + "bootstrap.php" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "description": "PHPStan - PHP Static Analysis Tool", + "keywords": [ + "dev", + "static analysis" + ], + "support": { + "docs": "https://phpstan.org/user-guide/getting-started", + "forum": "https://github.com/phpstan/phpstan/discussions", + "issues": "https://github.com/phpstan/phpstan/issues", + "security": "https://github.com/phpstan/phpstan/security/policy", + "source": "https://github.com/phpstan/phpstan-src" + }, + "funding": [ + { + "url": "https://github.com/ondrejmirtes", + "type": "github" + }, + { + "url": "https://github.com/phpstan", + "type": "github" + } + ], + "time": "2025-03-09T09:24:50+00:00" + }, + { + "name": "phpunit/php-code-coverage", + "version": "9.2.32", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/php-code-coverage.git", + "reference": "85402a822d1ecf1db1096959413d35e1c37cf1a5" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/php-code-coverage/zipball/85402a822d1ecf1db1096959413d35e1c37cf1a5", + "reference": "85402a822d1ecf1db1096959413d35e1c37cf1a5", + "shasum": "" + }, + "require": { + "ext-dom": "*", + "ext-libxml": "*", + "ext-xmlwriter": "*", + "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": "^9.6" + }, + "suggest": { + "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-main": "9.2.x-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 that provides collection, processing, and rendering functionality for PHP code coverage information.", + "homepage": "https://github.com/sebastianbergmann/php-code-coverage", + "keywords": [ + "coverage", + "testing", + "xunit" + ], + "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": "3.0.6", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/php-file-iterator.git", + "reference": "cf1c2e7c203ac650e352f4cc675a7021e7d1b3cf" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/php-file-iterator/zipball/cf1c2e7c203ac650e352f4cc675a7021e7d1b3cf", + "reference": "cf1c2e7c203ac650e352f4cc675a7021e7d1b3cf", + "shasum": "" + }, + "require": { + "php": ">=7.3" + }, + "require-dev": { + "phpunit/phpunit": "^9.3" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "3.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": "FilterIterator implementation that filters files based on a list of suffixes.", + "homepage": "https://github.com/sebastianbergmann/php-file-iterator/", + "keywords": [ + "filesystem", + "iterator" + ], + "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-invoker", + "version": "3.1.1", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/php-invoker.git", + "reference": "5a10147d0aaf65b58940a0b72f71c9ac0423cc67" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/php-invoker/zipball/5a10147d0aaf65b58940a0b72f71c9ac0423cc67", + "reference": "5a10147d0aaf65b58940a0b72f71c9ac0423cc67", + "shasum": "" + }, + "require": { + "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/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de", + "role": "lead" + } + ], + "description": "Simple template engine.", + "homepage": "https://github.com/sebastianbergmann/php-text-template/", + "keywords": [ + "template" + ], + "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": "5.0.3", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/php-timer.git", + "reference": "5a63ce20ed1b5bf577850e2c4e87f4aa902afbd2" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/php-timer/zipball/5a63ce20ed1b5bf577850e2c4e87f4aa902afbd2", + "reference": "5a63ce20ed1b5bf577850e2c4e87f4aa902afbd2", + "shasum": "" + }, + "require": { + "php": ">=7.3" + }, + "require-dev": { + "phpunit/phpunit": "^9.3" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "5.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": "Utility class for timing", + "homepage": "https://github.com/sebastianbergmann/php-timer/", + "keywords": [ + "timer" + ], + "support": { + "issues": "https://github.com/sebastianbergmann/php-timer/issues", + "source": "https://github.com/sebastianbergmann/php-timer/tree/5.0.3" + }, + "funding": [ + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + } + ], + "time": "2020-10-26T13:16:10+00:00" + }, + { + "name": "phpunit/phpunit", + "version": "9.6.22", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/phpunit.git", + "reference": "f80235cb4d3caa59ae09be3adf1ded27521d1a9c" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/phpunit/zipball/f80235cb4d3caa59ae09be3adf1ded27521d1a9c", + "reference": "f80235cb4d3caa59ae09be3adf1ded27521d1a9c", + "shasum": "" + }, + "require": { + "doctrine/instantiator": "^1.5.0 || ^2", + "ext-dom": "*", + "ext-json": "*", + "ext-libxml": "*", + "ext-mbstring": "*", + "ext-xml": "*", + "ext-xmlwriter": "*", + "myclabs/deep-copy": "^1.12.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-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" + ], + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "9.6-dev" + } + }, + "autoload": { + "files": [ + "src/Framework/Assert/Functions.php" + ], + "classmap": [ + "src/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de", + "role": "lead" + } + ], + "description": "The PHP Unit Testing framework.", + "homepage": "https://phpunit.de/", + "keywords": [ + "phpunit", + "testing", + "xunit" + ], + "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.22" + }, + "funding": [ + { + "url": "https://phpunit.de/sponsors.html", + "type": "custom" + }, + { + "url": "https://github.com/sebastianbergmann", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/phpunit/phpunit", + "type": "tidelift" + } + ], + "time": "2024-12-05T13:48:26+00:00" + }, + { + "name": "psr/container", + "version": "2.0.2", + "source": { + "type": "git", + "url": "https://github.com/php-fig/container.git", + "reference": "c71ecc56dfe541dbd90c5360474fbc405f8d5963" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/php-fig/container/zipball/c71ecc56dfe541dbd90c5360474fbc405f8d5963", + "reference": "c71ecc56dfe541dbd90c5360474fbc405f8d5963", + "shasum": "" + }, + "require": { + "php": ">=7.4.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "2.0.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Container\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "https://www.php-fig.org/" + } + ], + "description": "Common Container Interface (PHP FIG PSR-11)", + "homepage": "https://github.com/php-fig/container", + "keywords": [ + "PSR-11", + "container", + "container-interface", + "container-interop", + "psr" + ], + "support": { + "issues": "https://github.com/php-fig/container/issues", + "source": "https://github.com/php-fig/container/tree/2.0.2" + }, + "time": "2021-11-05T16:47:00+00:00" + }, + { + "name": "psr/event-dispatcher", + "version": "1.0.0", + "source": { + "type": "git", + "url": "https://github.com/php-fig/event-dispatcher.git", + "reference": "dbefd12671e8a14ec7f180cab83036ed26714bb0" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/php-fig/event-dispatcher/zipball/dbefd12671e8a14ec7f180cab83036ed26714bb0", + "reference": "dbefd12671e8a14ec7f180cab83036ed26714bb0", + "shasum": "" + }, + "require": { + "php": ">=7.2.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.0.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\EventDispatcher\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "http://www.php-fig.org/" + } + ], + "description": "Standard interfaces for event handling.", + "keywords": [ + "events", + "psr", + "psr-14" + ], + "support": { + "issues": "https://github.com/php-fig/event-dispatcher/issues", + "source": "https://github.com/php-fig/event-dispatcher/tree/1.0.0" + }, + "time": "2019-01-08T18:20:26+00:00" + }, + { + "name": "sebastian/cli-parser", + "version": "1.0.2", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/cli-parser.git", + "reference": "2b56bea83a09de3ac06bb18b92f068e60cc6f50b" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/cli-parser/zipball/2b56bea83a09de3ac06bb18b92f068e60cc6f50b", + "reference": "2b56bea83a09de3ac06bb18b92f068e60cc6f50b", + "shasum": "" + }, + "require": { + "php": ">=7.3" + }, + "require-dev": { + "phpunit/phpunit": "^9.3" + }, + "type": "library", + "extra": { + "branch-alias": { + "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 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": "2024-03-02T06:27:43+00:00" + }, + { + "name": "sebastian/code-unit", + "version": "1.0.8", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/code-unit.git", + "reference": "1fc9f64c0927627ef78ba436c9b17d967e68e120" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/code-unit/zipball/1fc9f64c0927627ef78ba436c9b17d967e68e120", + "reference": "1fc9f64c0927627ef78ba436c9b17d967e68e120", + "shasum": "" + }, + "require": { + "php": ">=7.3" + }, + "require-dev": { + "phpunit/phpunit": "^9.3" + }, + "type": "library", + "extra": { + "branch-alias": { + "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": { + "classmap": [ + "src/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de" + } + ], + "description": "Looks up which function or method a line of code belongs to", + "homepage": "https://github.com/sebastianbergmann/code-unit-reverse-lookup/", + "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": "4.0.8", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/comparator.git", + "reference": "fa0f136dd2334583309d32b62544682ee972b51a" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/comparator/zipball/fa0f136dd2334583309d32b62544682ee972b51a", + "reference": "fa0f136dd2334583309d32b62544682ee972b51a", + "shasum": "" + }, + "require": { + "php": ">=7.3", + "sebastian/diff": "^4.0", + "sebastian/exporter": "^4.0" + }, + "require-dev": { + "phpunit/phpunit": "^9.3" + }, + "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": "Jeff Welch", + "email": "whatthejeff@gmail.com" + }, + { + "name": "Volker Dusch", + "email": "github@wallbash.com" + }, + { + "name": "Bernhard Schussek", + "email": "bschussek@2bepublished.at" + } + ], + "description": "Provides the functionality to compare PHP values for equality", + "homepage": "https://github.com/sebastianbergmann/comparator", + "keywords": [ + "comparator", + "compare", + "equality" + ], + "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/complexity", + "version": "2.0.3", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/complexity.git", + "reference": "25f207c40d62b8b7aa32f5ab026c53561964053a" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/complexity/zipball/25f207c40d62b8b7aa32f5ab026c53561964053a", + "reference": "25f207c40d62b8b7aa32f5ab026c53561964053a", + "shasum": "" + }, + "require": { + "nikic/php-parser": "^4.18 || ^5.0", + "php": ">=7.3" + }, + "require-dev": { + "phpunit/phpunit": "^9.3" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "2.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 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", + "udiff", + "unidiff", + "unified diff" + ], + "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": "5.1.5", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/environment.git", + "reference": "830c43a844f1f8d5b7a1f6d6076b784454d8b7ed" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/environment/zipball/830c43a844f1f8d5b7a1f6d6076b784454d8b7ed", + "reference": "830c43a844f1f8d5b7a1f6d6076b784454d8b7ed", + "shasum": "" + }, + "require": { + "php": ">=7.3" + }, + "require-dev": { + "phpunit/phpunit": "^9.3" + }, + "suggest": { + "ext-posix": "*" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "5.1-dev" + } + }, + "autoload": { + "classmap": [ + "src/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de" + } + ], + "description": "Provides functionality to handle HHVM/PHP environments", + "homepage": "http://www.github.com/sebastianbergmann/environment", + "keywords": [ + "Xdebug", + "environment", + "hhvm" + ], + "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": "4.0.6", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/exporter.git", + "reference": "78c00df8f170e02473b682df15bfcdacc3d32d72" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/exporter/zipball/78c00df8f170e02473b682df15bfcdacc3d32d72", + "reference": "78c00df8f170e02473b682df15bfcdacc3d32d72", + "shasum": "" + }, + "require": { + "php": ">=7.3", + "sebastian/recursion-context": "^4.0" + }, + "require-dev": { + "ext-mbstring": "*", + "phpunit/phpunit": "^9.3" + }, + "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": "Jeff Welch", + "email": "whatthejeff@gmail.com" + }, + { + "name": "Volker Dusch", + "email": "github@wallbash.com" + }, + { + "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": "https://www.github.com/sebastianbergmann/exporter", + "keywords": [ + "export", + "exporter" + ], + "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": "5.0.7", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/global-state.git", + "reference": "bca7df1f32ee6fe93b4d4a9abbf69e13a4ada2c9" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/global-state/zipball/bca7df1f32ee6fe93b4d4a9abbf69e13a4ada2c9", + "reference": "bca7df1f32ee6fe93b4d4a9abbf69e13a4ada2c9", + "shasum": "" + }, + "require": { + "php": ">=7.3", + "sebastian/object-reflector": "^2.0", + "sebastian/recursion-context": "^4.0" + }, + "require-dev": { + "ext-dom": "*", + "phpunit/phpunit": "^9.3" + }, + "suggest": { + "ext-uopz": "*" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "5.0-dev" + } + }, + "autoload": { + "classmap": [ + "src/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de" + } + ], + "description": "Snapshotting of global state", + "homepage": "http://www.github.com/sebastianbergmann/global-state", + "keywords": [ + "global state" + ], + "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/lines-of-code", + "version": "1.0.4", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/lines-of-code.git", + "reference": "e1e4a170560925c26d424b6a03aed157e7dcc5c5" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/lines-of-code/zipball/e1e4a170560925c26d424b6a03aed157e7dcc5c5", + "reference": "e1e4a170560925c26d424b6a03aed157e7dcc5c5", + "shasum": "" + }, + "require": { + "nikic/php-parser": "^4.18 || ^5.0", + "php": ">=7.3" + }, + "require-dev": { + "phpunit/phpunit": "^9.3" + }, + "type": "library", + "extra": { + "branch-alias": { + "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": { + "classmap": [ + "src/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de" + } + ], + "description": "Traverses array structures and object graphs to enumerate all referenced objects", + "homepage": "https://github.com/sebastianbergmann/object-enumerator/", + "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": "2.0.4", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/object-reflector.git", + "reference": "b4f479ebdbf63ac605d183ece17d8d7fe49c15c7" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/object-reflector/zipball/b4f479ebdbf63ac605d183ece17d8d7fe49c15c7", + "reference": "b4f479ebdbf63ac605d183ece17d8d7fe49c15c7", + "shasum": "" + }, + "require": { + "php": ">=7.3" + }, + "require-dev": { + "phpunit/phpunit": "^9.3" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "2.0-dev" + } + }, + "autoload": { + "classmap": [ + "src/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de" + } + ], + "description": "Allows reflection of object attributes, including inherited and non-public ones", + "homepage": "https://github.com/sebastianbergmann/object-reflector/", + "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": "4.0.5", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/recursion-context.git", + "reference": "e75bd0f07204fec2a0af9b0f3cfe97d05f92efc1" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/recursion-context/zipball/e75bd0f07204fec2a0af9b0f3cfe97d05f92efc1", + "reference": "e75bd0f07204fec2a0af9b0f3cfe97d05f92efc1", + "shasum": "" + }, + "require": { + "php": ">=7.3" + }, + "require-dev": { + "phpunit/phpunit": "^9.3" + }, + "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": "Jeff Welch", + "email": "whatthejeff@gmail.com" + }, + { + "name": "Adam Harvey", + "email": "aharvey@php.net" + } + ], + "description": "Provides functionality to recursively process PHP variables", + "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": "3.0.4", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/resource-operations.git", + "reference": "05d5692a7993ecccd56a03e40cd7e5b09b1d404e" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/resource-operations/zipball/05d5692a7993ecccd56a03e40cd7e5b09b1d404e", + "reference": "05d5692a7993ecccd56a03e40cd7e5b09b1d404e", + "shasum": "" + }, + "require": { + "php": ">=7.3" + }, + "require-dev": { + "phpunit/phpunit": "^9.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "3.0-dev" + } + }, + "autoload": { + "classmap": [ + "src/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de" + } + ], + "description": "Provides a list of PHP built-in functions that operate on resources", + "homepage": "https://www.github.com/sebastianbergmann/resource-operations", + "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/type", + "version": "3.2.1", + "source": { + "type": "git", + "url": "https://github.com/sebastianbergmann/type.git", + "reference": "75e2c2a32f5e0b3aef905b9ed0b179b953b3d7c7" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/sebastianbergmann/type/zipball/75e2c2a32f5e0b3aef905b9ed0b179b953b3d7c7", + "reference": "75e2c2a32f5e0b3aef905b9ed0b179b953b3d7c7", + "shasum": "" + }, + "require": { + "php": ">=7.3" + }, + "require-dev": { + "phpunit/phpunit": "^9.5" + }, + "type": "library", + "extra": { + "branch-alias": { + "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": { + "classmap": [ + "src/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Sebastian Bergmann", + "email": "sebastian@phpunit.de", + "role": "lead" + } + ], + "description": "Library that helps with managing the version number of Git-hosted PHP projects", + "homepage": "https://github.com/sebastianbergmann/version", + "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": "squizlabs/php_codesniffer", + "version": "3.11.3", + "source": { + "type": "git", + "url": "https://github.com/PHPCSStandards/PHP_CodeSniffer.git", + "reference": "ba05f990e79cbe69b9f35c8c1ac8dca7eecc3a10" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/PHPCSStandards/PHP_CodeSniffer/zipball/ba05f990e79cbe69b9f35c8c1ac8dca7eecc3a10", + "reference": "ba05f990e79cbe69b9f35c8c1ac8dca7eecc3a10", + "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/phpcsstandards", + "type": "thanks_dev" + } + ], + "time": "2025-01-23T17:04:15+00:00" + }, + { + "name": "symfony/config", + "version": "v6.4.14", + "source": { + "type": "git", + "url": "https://github.com/symfony/config.git", + "reference": "4e55e7e4ffddd343671ea972216d4509f46c22ef" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/config/zipball/4e55e7e4ffddd343671ea972216d4509f46c22ef", + "reference": "4e55e7e4ffddd343671ea972216d4509f46c22ef", + "shasum": "" + }, + "require": { + "php": ">=8.1", + "symfony/deprecation-contracts": "^2.5|^3", + "symfony/filesystem": "^5.4|^6.0|^7.0", + "symfony/polyfill-ctype": "~1.8" + }, + "conflict": { + "symfony/finder": "<5.4", + "symfony/service-contracts": "<2.5" + }, + "require-dev": { + "symfony/event-dispatcher": "^5.4|^6.0|^7.0", + "symfony/finder": "^5.4|^6.0|^7.0", + "symfony/messenger": "^5.4|^6.0|^7.0", + "symfony/service-contracts": "^2.5|^3", + "symfony/yaml": "^5.4|^6.0|^7.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Symfony\\Component\\Config\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Fabien Potencier", + "email": "fabien@symfony.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Helps you find, load, combine, autofill and validate configuration values of any kind", + "homepage": "https://symfony.com", + "support": { + "source": "https://github.com/symfony/config/tree/v6.4.14" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2024-11-04T11:33:53+00:00" + }, + { + "name": "symfony/console", + "version": "v6.4.17", + "source": { + "type": "git", + "url": "https://github.com/symfony/console.git", + "reference": "799445db3f15768ecc382ac5699e6da0520a0a04" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/console/zipball/799445db3f15768ecc382ac5699e6da0520a0a04", + "reference": "799445db3f15768ecc382ac5699e6da0520a0a04", + "shasum": "" + }, + "require": { + "php": ">=8.1", + "symfony/deprecation-contracts": "^2.5|^3", + "symfony/polyfill-mbstring": "~1.0", + "symfony/service-contracts": "^2.5|^3", + "symfony/string": "^5.4|^6.0|^7.0" + }, + "conflict": { + "symfony/dependency-injection": "<5.4", + "symfony/dotenv": "<5.4", + "symfony/event-dispatcher": "<5.4", + "symfony/lock": "<5.4", + "symfony/process": "<5.4" + }, + "provide": { + "psr/log-implementation": "1.0|2.0|3.0" + }, + "require-dev": { + "psr/log": "^1|^2|^3", + "symfony/config": "^5.4|^6.0|^7.0", + "symfony/dependency-injection": "^5.4|^6.0|^7.0", + "symfony/event-dispatcher": "^5.4|^6.0|^7.0", + "symfony/http-foundation": "^6.4|^7.0", + "symfony/http-kernel": "^6.4|^7.0", + "symfony/lock": "^5.4|^6.0|^7.0", + "symfony/messenger": "^5.4|^6.0|^7.0", + "symfony/process": "^5.4|^6.0|^7.0", + "symfony/stopwatch": "^5.4|^6.0|^7.0", + "symfony/var-dumper": "^5.4|^6.0|^7.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Symfony\\Component\\Console\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Fabien Potencier", + "email": "fabien@symfony.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Eases the creation of beautiful and testable command line interfaces", + "homepage": "https://symfony.com", + "keywords": [ + "cli", + "command-line", + "console", + "terminal" + ], + "support": { + "source": "https://github.com/symfony/console/tree/v6.4.17" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2024-12-07T12:07:30+00:00" + }, + { + "name": "symfony/dependency-injection", + "version": "v6.4.19", + "source": { + "type": "git", + "url": "https://github.com/symfony/dependency-injection.git", + "reference": "b343c3b2f1539fe41331657b37d5c96c1d1ea842" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/dependency-injection/zipball/b343c3b2f1539fe41331657b37d5c96c1d1ea842", + "reference": "b343c3b2f1539fe41331657b37d5c96c1d1ea842", + "shasum": "" + }, + "require": { + "php": ">=8.1", + "psr/container": "^1.1|^2.0", + "symfony/deprecation-contracts": "^2.5|^3", + "symfony/service-contracts": "^2.5|^3.0", + "symfony/var-exporter": "^6.2.10|^7.0" + }, + "conflict": { + "ext-psr": "<1.1|>=2", + "symfony/config": "<6.1", + "symfony/finder": "<5.4", + "symfony/proxy-manager-bridge": "<6.3", + "symfony/yaml": "<5.4" + }, + "provide": { + "psr/container-implementation": "1.1|2.0", + "symfony/service-implementation": "1.1|2.0|3.0" + }, + "require-dev": { + "symfony/config": "^6.1|^7.0", + "symfony/expression-language": "^5.4|^6.0|^7.0", + "symfony/yaml": "^5.4|^6.0|^7.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Symfony\\Component\\DependencyInjection\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Fabien Potencier", + "email": "fabien@symfony.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Allows you to standardize and centralize the way objects are constructed in your application", + "homepage": "https://symfony.com", + "support": { + "source": "https://github.com/symfony/dependency-injection/tree/v6.4.19" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2025-02-20T10:02:49+00:00" + }, + { + "name": "symfony/event-dispatcher", + "version": "v6.4.13", + "source": { + "type": "git", + "url": "https://github.com/symfony/event-dispatcher.git", + "reference": "0ffc48080ab3e9132ea74ef4e09d8dcf26bf897e" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/event-dispatcher/zipball/0ffc48080ab3e9132ea74ef4e09d8dcf26bf897e", + "reference": "0ffc48080ab3e9132ea74ef4e09d8dcf26bf897e", + "shasum": "" + }, + "require": { + "php": ">=8.1", + "symfony/event-dispatcher-contracts": "^2.5|^3" + }, + "conflict": { + "symfony/dependency-injection": "<5.4", + "symfony/service-contracts": "<2.5" + }, + "provide": { + "psr/event-dispatcher-implementation": "1.0", + "symfony/event-dispatcher-implementation": "2.0|3.0" + }, + "require-dev": { + "psr/log": "^1|^2|^3", + "symfony/config": "^5.4|^6.0|^7.0", + "symfony/dependency-injection": "^5.4|^6.0|^7.0", + "symfony/error-handler": "^5.4|^6.0|^7.0", + "symfony/expression-language": "^5.4|^6.0|^7.0", + "symfony/http-foundation": "^5.4|^6.0|^7.0", + "symfony/service-contracts": "^2.5|^3", + "symfony/stopwatch": "^5.4|^6.0|^7.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Symfony\\Component\\EventDispatcher\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Fabien Potencier", + "email": "fabien@symfony.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Provides tools that allow your application components to communicate with each other by dispatching events and listening to them", + "homepage": "https://symfony.com", + "support": { + "source": "https://github.com/symfony/event-dispatcher/tree/v6.4.13" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2024-09-25T14:18:03+00:00" + }, + { + "name": "symfony/event-dispatcher-contracts", + "version": "v3.5.1", + "source": { + "type": "git", + "url": "https://github.com/symfony/event-dispatcher-contracts.git", + "reference": "7642f5e970b672283b7823222ae8ef8bbc160b9f" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/event-dispatcher-contracts/zipball/7642f5e970b672283b7823222ae8ef8bbc160b9f", + "reference": "7642f5e970b672283b7823222ae8ef8bbc160b9f", + "shasum": "" + }, + "require": { + "php": ">=8.1", + "psr/event-dispatcher": "^1" + }, + "type": "library", + "extra": { + "thanks": { + "url": "https://github.com/symfony/contracts", + "name": "symfony/contracts" + }, + "branch-alias": { + "dev-main": "3.5-dev" + } + }, + "autoload": { + "psr-4": { + "Symfony\\Contracts\\EventDispatcher\\": "" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Generic abstractions related to dispatching event", + "homepage": "https://symfony.com", + "keywords": [ + "abstractions", + "contracts", + "decoupling", + "interfaces", + "interoperability", + "standards" + ], + "support": { + "source": "https://github.com/symfony/event-dispatcher-contracts/tree/v3.5.1" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2024-09-25T14:20:29+00:00" + }, + { + "name": "symfony/filesystem", + "version": "v6.4.13", + "source": { + "type": "git", + "url": "https://github.com/symfony/filesystem.git", + "reference": "4856c9cf585d5a0313d8d35afd681a526f038dd3" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/filesystem/zipball/4856c9cf585d5a0313d8d35afd681a526f038dd3", + "reference": "4856c9cf585d5a0313d8d35afd681a526f038dd3", + "shasum": "" + }, + "require": { + "php": ">=8.1", + "symfony/polyfill-ctype": "~1.8", + "symfony/polyfill-mbstring": "~1.8" + }, + "require-dev": { + "symfony/process": "^5.4|^6.4|^7.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Symfony\\Component\\Filesystem\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Fabien Potencier", + "email": "fabien@symfony.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Provides basic utilities for the filesystem", + "homepage": "https://symfony.com", + "support": { + "source": "https://github.com/symfony/filesystem/tree/v6.4.13" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2024-10-25T15:07:50+00:00" + }, + { + "name": "symfony/polyfill-ctype", + "version": "v1.31.0", + "source": { + "type": "git", + "url": "https://github.com/symfony/polyfill-ctype.git", + "reference": "a3cc8b044a6ea513310cbd48ef7333b384945638" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/polyfill-ctype/zipball/a3cc8b044a6ea513310cbd48ef7333b384945638", + "reference": "a3cc8b044a6ea513310cbd48ef7333b384945638", + "shasum": "" + }, + "require": { + "php": ">=7.2" + }, + "provide": { + "ext-ctype": "*" + }, + "suggest": { + "ext-ctype": "For best performance" + }, + "type": "library", + "extra": { + "thanks": { + "url": "https://github.com/symfony/polyfill", + "name": "symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Ctype\\": "" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Gert de Pagter", + "email": "BackEndTea@gmail.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Symfony polyfill for ctype functions", + "homepage": "https://symfony.com", + "keywords": [ + "compatibility", + "ctype", + "polyfill", + "portable" + ], + "support": { + "source": "https://github.com/symfony/polyfill-ctype/tree/v1.31.0" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2024-09-09T11:45:10+00:00" + }, + { + "name": "symfony/polyfill-intl-grapheme", + "version": "v1.31.0", + "source": { + "type": "git", + "url": "https://github.com/symfony/polyfill-intl-grapheme.git", + "reference": "b9123926e3b7bc2f98c02ad54f6a4b02b91a8abe" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/polyfill-intl-grapheme/zipball/b9123926e3b7bc2f98c02ad54f6a4b02b91a8abe", + "reference": "b9123926e3b7bc2f98c02ad54f6a4b02b91a8abe", + "shasum": "" + }, + "require": { + "php": ">=7.2" + }, + "suggest": { + "ext-intl": "For best performance" + }, + "type": "library", + "extra": { + "thanks": { + "url": "https://github.com/symfony/polyfill", + "name": "symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Intl\\Grapheme\\": "" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Symfony polyfill for intl's grapheme_* functions", + "homepage": "https://symfony.com", + "keywords": [ + "compatibility", + "grapheme", + "intl", + "polyfill", + "portable", + "shim" + ], + "support": { + "source": "https://github.com/symfony/polyfill-intl-grapheme/tree/v1.31.0" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2024-09-09T11:45:10+00:00" + }, + { + "name": "symfony/polyfill-intl-normalizer", + "version": "v1.31.0", + "source": { + "type": "git", + "url": "https://github.com/symfony/polyfill-intl-normalizer.git", + "reference": "3833d7255cc303546435cb650316bff708a1c75c" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/polyfill-intl-normalizer/zipball/3833d7255cc303546435cb650316bff708a1c75c", + "reference": "3833d7255cc303546435cb650316bff708a1c75c", + "shasum": "" + }, + "require": { + "php": ">=7.2" + }, + "suggest": { + "ext-intl": "For best performance" + }, + "type": "library", + "extra": { + "thanks": { + "url": "https://github.com/symfony/polyfill", + "name": "symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Intl\\Normalizer\\": "" + }, + "classmap": [ + "Resources/stubs" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Symfony polyfill for intl's Normalizer class and related functions", + "homepage": "https://symfony.com", + "keywords": [ + "compatibility", + "intl", + "normalizer", + "polyfill", + "portable", + "shim" + ], + "support": { + "source": "https://github.com/symfony/polyfill-intl-normalizer/tree/v1.31.0" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2024-09-09T11:45:10+00:00" + }, + { + "name": "symfony/polyfill-mbstring", + "version": "v1.31.0", + "source": { + "type": "git", + "url": "https://github.com/symfony/polyfill-mbstring.git", + "reference": "85181ba99b2345b0ef10ce42ecac37612d9fd341" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/polyfill-mbstring/zipball/85181ba99b2345b0ef10ce42ecac37612d9fd341", + "reference": "85181ba99b2345b0ef10ce42ecac37612d9fd341", + "shasum": "" + }, + "require": { + "php": ">=7.2" + }, + "provide": { + "ext-mbstring": "*" + }, + "suggest": { + "ext-mbstring": "For best performance" + }, + "type": "library", + "extra": { + "thanks": { + "url": "https://github.com/symfony/polyfill", + "name": "symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Mbstring\\": "" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Symfony polyfill for the Mbstring extension", + "homepage": "https://symfony.com", + "keywords": [ + "compatibility", + "mbstring", + "polyfill", + "portable", + "shim" + ], + "support": { + "source": "https://github.com/symfony/polyfill-mbstring/tree/v1.31.0" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2024-09-09T11:45:10+00:00" + }, + { + "name": "symfony/service-contracts", + "version": "v3.5.1", + "source": { + "type": "git", + "url": "https://github.com/symfony/service-contracts.git", + "reference": "e53260aabf78fb3d63f8d79d69ece59f80d5eda0" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/service-contracts/zipball/e53260aabf78fb3d63f8d79d69ece59f80d5eda0", + "reference": "e53260aabf78fb3d63f8d79d69ece59f80d5eda0", + "shasum": "" + }, + "require": { + "php": ">=8.1", + "psr/container": "^1.1|^2.0", + "symfony/deprecation-contracts": "^2.5|^3" + }, + "conflict": { + "ext-psr": "<1.1|>=2" + }, + "type": "library", + "extra": { + "thanks": { + "url": "https://github.com/symfony/contracts", + "name": "symfony/contracts" + }, + "branch-alias": { + "dev-main": "3.5-dev" + } + }, + "autoload": { + "psr-4": { + "Symfony\\Contracts\\Service\\": "" + }, + "exclude-from-classmap": [ + "/Test/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Generic abstractions related to writing services", + "homepage": "https://symfony.com", + "keywords": [ + "abstractions", + "contracts", + "decoupling", + "interfaces", + "interoperability", + "standards" + ], + "support": { + "source": "https://github.com/symfony/service-contracts/tree/v3.5.1" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2024-09-25T14:20:29+00:00" + }, + { + "name": "symfony/string", + "version": "v6.4.15", + "source": { + "type": "git", + "url": "https://github.com/symfony/string.git", + "reference": "73a5e66ea2e1677c98d4449177c5a9cf9d8b4c6f" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/string/zipball/73a5e66ea2e1677c98d4449177c5a9cf9d8b4c6f", + "reference": "73a5e66ea2e1677c98d4449177c5a9cf9d8b4c6f", + "shasum": "" + }, + "require": { + "php": ">=8.1", + "symfony/polyfill-ctype": "~1.8", + "symfony/polyfill-intl-grapheme": "~1.0", + "symfony/polyfill-intl-normalizer": "~1.0", + "symfony/polyfill-mbstring": "~1.0" + }, + "conflict": { + "symfony/translation-contracts": "<2.5" + }, + "require-dev": { + "symfony/error-handler": "^5.4|^6.0|^7.0", + "symfony/http-client": "^5.4|^6.0|^7.0", + "symfony/intl": "^6.2|^7.0", + "symfony/translation-contracts": "^2.5|^3.0", + "symfony/var-exporter": "^5.4|^6.0|^7.0" + }, + "type": "library", + "autoload": { + "files": [ + "Resources/functions.php" + ], + "psr-4": { + "Symfony\\Component\\String\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Provides an object-oriented API to strings and deals with bytes, UTF-8 code points and grapheme clusters in a unified way", + "homepage": "https://symfony.com", + "keywords": [ + "grapheme", + "i18n", + "string", + "unicode", + "utf-8", + "utf8" + ], + "support": { + "source": "https://github.com/symfony/string/tree/v6.4.15" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2024-11-13T13:31:12+00:00" + }, + { + "name": "symfony/translation", + "version": "v6.4.19", + "source": { + "type": "git", + "url": "https://github.com/symfony/translation.git", + "reference": "3b9bf9f33997c064885a7bfc126c14b9daa0e00e" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/translation/zipball/3b9bf9f33997c064885a7bfc126c14b9daa0e00e", + "reference": "3b9bf9f33997c064885a7bfc126c14b9daa0e00e", + "shasum": "" + }, + "require": { + "php": ">=8.1", + "symfony/deprecation-contracts": "^2.5|^3", + "symfony/polyfill-mbstring": "~1.0", + "symfony/translation-contracts": "^2.5|^3.0" + }, + "conflict": { + "symfony/config": "<5.4", + "symfony/console": "<5.4", + "symfony/dependency-injection": "<5.4", + "symfony/http-client-contracts": "<2.5", + "symfony/http-kernel": "<5.4", + "symfony/service-contracts": "<2.5", + "symfony/twig-bundle": "<5.4", + "symfony/yaml": "<5.4" + }, + "provide": { + "symfony/translation-implementation": "2.3|3.0" + }, + "require-dev": { + "nikic/php-parser": "^4.18|^5.0", + "psr/log": "^1|^2|^3", + "symfony/config": "^5.4|^6.0|^7.0", + "symfony/console": "^5.4|^6.0|^7.0", + "symfony/dependency-injection": "^5.4|^6.0|^7.0", + "symfony/finder": "^5.4|^6.0|^7.0", + "symfony/http-client-contracts": "^2.5|^3.0", + "symfony/http-kernel": "^5.4|^6.0|^7.0", + "symfony/intl": "^5.4|^6.0|^7.0", + "symfony/polyfill-intl-icu": "^1.21", + "symfony/routing": "^5.4|^6.0|^7.0", + "symfony/service-contracts": "^2.5|^3", + "symfony/yaml": "^5.4|^6.0|^7.0" + }, + "type": "library", + "autoload": { + "files": [ + "Resources/functions.php" + ], + "psr-4": { + "Symfony\\Component\\Translation\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Fabien Potencier", + "email": "fabien@symfony.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Provides tools to internationalize your application", + "homepage": "https://symfony.com", + "support": { + "source": "https://github.com/symfony/translation/tree/v6.4.19" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2025-02-13T10:18:43+00:00" + }, + { + "name": "symfony/translation-contracts", + "version": "v3.5.1", + "source": { + "type": "git", + "url": "https://github.com/symfony/translation-contracts.git", + "reference": "4667ff3bd513750603a09c8dedbea942487fb07c" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/translation-contracts/zipball/4667ff3bd513750603a09c8dedbea942487fb07c", + "reference": "4667ff3bd513750603a09c8dedbea942487fb07c", + "shasum": "" + }, + "require": { + "php": ">=8.1" + }, + "type": "library", + "extra": { + "thanks": { + "url": "https://github.com/symfony/contracts", + "name": "symfony/contracts" + }, + "branch-alias": { + "dev-main": "3.5-dev" + } + }, + "autoload": { + "psr-4": { + "Symfony\\Contracts\\Translation\\": "" + }, + "exclude-from-classmap": [ + "/Test/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Generic abstractions related to translation", + "homepage": "https://symfony.com", + "keywords": [ + "abstractions", + "contracts", + "decoupling", + "interfaces", + "interoperability", + "standards" + ], + "support": { + "source": "https://github.com/symfony/translation-contracts/tree/v3.5.1" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2024-09-25T14:20:29+00:00" + }, + { + "name": "symfony/var-exporter", + "version": "v6.4.19", + "source": { + "type": "git", + "url": "https://github.com/symfony/var-exporter.git", + "reference": "be6e71b0c257884c1107313de5d247741cfea172" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/var-exporter/zipball/be6e71b0c257884c1107313de5d247741cfea172", + "reference": "be6e71b0c257884c1107313de5d247741cfea172", + "shasum": "" + }, + "require": { + "php": ">=8.1", + "symfony/deprecation-contracts": "^2.5|^3" + }, + "require-dev": { + "symfony/property-access": "^6.4|^7.0", + "symfony/serializer": "^6.4|^7.0", + "symfony/var-dumper": "^5.4|^6.0|^7.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Symfony\\Component\\VarExporter\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Allows exporting any serializable PHP data structure to plain PHP code", + "homepage": "https://symfony.com", + "keywords": [ + "clone", + "construct", + "export", + "hydrate", + "instantiate", + "lazy-loading", + "proxy", + "serialize" + ], + "support": { + "source": "https://github.com/symfony/var-exporter/tree/v6.4.19" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2025-02-13T09:33:32+00:00" + }, + { + "name": "symfony/yaml", + "version": "v6.4.18", + "source": { + "type": "git", + "url": "https://github.com/symfony/yaml.git", + "reference": "bf598c9d9bb4a22f495a4e26e4c4fce2f8ecefc5" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/yaml/zipball/bf598c9d9bb4a22f495a4e26e4c4fce2f8ecefc5", + "reference": "bf598c9d9bb4a22f495a4e26e4c4fce2f8ecefc5", + "shasum": "" + }, + "require": { + "php": ">=8.1", + "symfony/deprecation-contracts": "^2.5|^3", + "symfony/polyfill-ctype": "^1.8" + }, + "conflict": { + "symfony/console": "<5.4" + }, + "require-dev": { + "symfony/console": "^5.4|^6.0|^7.0" + }, + "bin": [ + "Resources/bin/yaml-lint" + ], + "type": "library", + "autoload": { + "psr-4": { + "Symfony\\Component\\Yaml\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Fabien Potencier", + "email": "fabien@symfony.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Loads and dumps YAML files", + "homepage": "https://symfony.com", + "support": { + "source": "https://github.com/symfony/yaml/tree/v6.4.18" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2025-01-07T09:44:41+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.2 || ^8.0" + }, + "type": "library", + "autoload": { + "classmap": [ + "src/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Arne Blankerts", + "email": "arne@blankerts.de", + "role": "Developer" + } + ], + "description": "A small library for converting tokenized PHP source code into XML and potentially other formats", + "support": { + "issues": "https://github.com/theseer/tokenizer/issues", + "source": "https://github.com/theseer/tokenizer/tree/1.2.3" + }, + "funding": [ + { + "url": "https://github.com/theseer", + "type": "github" + } + ], + "time": "2024-03-03T12:36:25+00:00" + } + ], + "aliases": [], + "minimum-stability": "stable", + "stability-flags": {}, + "prefer-stable": false, + "prefer-lowest": false, + "platform": { + "php": ">=8.1" + }, + "platform-dev": {}, + "plugin-api-version": "2.6.0" +} diff --git a/examples/Membersips/channel_members.php b/examples/Membersips/channel_members.php new file mode 100644 index 00000000..bf989a6c --- /dev/null +++ b/examples/Membersips/channel_members.php @@ -0,0 +1,37 @@ +setPublishKey('demo'); +$pnConfig->setSubscribeKey('demo'); +$pnConfig->setUserId('demo'); + +$pubnub = new PubNub($pnConfig); + +$includes = new PNMemberIncludes(); +$includes->user()->userId()->userCustom()->userType()->userStatus()->custom()->status()->type(); + +$channel_members = [ + (new PNChannelMember('uuid1')) + ->setCustom(["a" => "b"]) + ->setStatus("status") + ->setType("type"), + (new PNChannelMember('uuid1')) + ->setCustom(["c" => "d"]) + ->setStatus("status") + ->setType("type") +]; + +$set_response = $pubnub->setMembers()->include($includes)->channel("ch")->members($channel_members)->sync(); + +var_dump($set_response); diff --git a/examples/customClient.php b/examples/customClient.php new file mode 100644 index 00000000..506bc967 --- /dev/null +++ b/examples/customClient.php @@ -0,0 +1,246 @@ +statusCode = $statusCode; + $this->headers = $headers; + $this->body = new Stream($body); + } + + public function getStatusCode(): int + { + return $this->statusCode; + } + + public function getReasonPhrase(): string + { + return ''; + } + + public function getProtocolVersion(): string + { + return ''; + } + + public function getHeaders(): array + { + return $this->headers; + } + + public function getBody(): StreamInterface + { + return $this->body; + } + + public function withStatus($code, $reasonPhrase = ''): self + { + $this->statusCode = $code; + return $this; + } + + public function withHeader($name, $value): self + { + $this->headers[$name] = $value; + return $this; + } + + public function withAddedHeader($name, $value): self + { + $this->headers[$name][] = $value; + return $this; + } + + public function withoutHeader($name): self + { + unset($this->headers[$name]); + return $this; + } + + public function withProtocolVersion($version): self + { + return $this; + } + + public function hasHeader($name): bool + { + return isset($this->headers[$name]); + } + + public function getHeader($name): array + { + return $this->headers[$name] ?? []; + } + + public function getHeaderLine($name): string + { + return implode(', ', $this->headers[$name] ?? []); + } + + public function withBody(StreamInterface $body): self + { + $this->body = $body; + return $this; + } +} + +class Stream implements StreamInterface +{ + private $stream; + private $size; + + public function __construct(string $content = '') + { + $this->stream = fopen('php://temp', 'r+'); + fwrite($this->stream, $content); + rewind($this->stream); + $this->size = strlen($content); + } + + public function __toString(): string + { + return stream_get_contents($this->stream, -1, 0); + } + + public function close(): void + { + fclose($this->stream); + } + + public function detach() + { + $result = $this->stream; + $this->stream = null; + return $result; + } + + public function getSize(): ?int + { + return $this->size; + } + + public function tell(): int + { + return ftell($this->stream); + } + + public function eof(): bool + { + return feof($this->stream); + } + + public function isSeekable(): bool + { + return true; + } + + public function seek($offset, $whence = SEEK_SET): void + { + fseek($this->stream, $offset, $whence); + } + + public function rewind(): void + { + rewind($this->stream); + } + + public function isWritable(): bool + { + return true; + } + + public function write($string): int + { + $result = fwrite($this->stream, $string); + $this->size += strlen($string); + return $result; + } + + public function isReadable(): bool + { + return true; + } + + public function read($length): string + { + return fread($this->stream, $length); + } + + public function getContents(): string + { + return stream_get_contents($this->stream); + } + + public function getMetadata($key = null) + { + $meta = stream_get_meta_data($this->stream); + if ($key === null) { + return $meta; + } + return $meta[$key] ?? null; + } +} + +class CustomClient implements ClientInterface +{ + public function sendRequest(RequestInterface $request): ResponseInterface + { + $ch = curl_init(); + + curl_setopt($ch, CURLOPT_URL, (string) $request->getUri()); + curl_setopt($ch, CURLOPT_CUSTOMREQUEST, $request->getMethod()); + curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); + + $headers = []; + foreach ($request->getHeaders() as $name => $values) { + foreach ($values as $value) { + $headers[] = $name . ': ' . $value; + } + } + curl_setopt($ch, CURLOPT_HTTPHEADER, $headers); + + if ($request->getBody()->getSize() > 0) { + curl_setopt($ch, CURLOPT_POSTFIELDS, (string) $request->getBody()); + } + print("\n doing request..."); + $responseBody = curl_exec($ch); + print("done\n\n"); + $responseCode = curl_getinfo($ch, CURLINFO_HTTP_CODE); + + if ($responseBody === false) { + throw new \RuntimeException('Curl error: ' . curl_error($ch)); + } + + curl_close($ch); + + return new Response($responseCode, [], $responseBody); + } +} + +$config = new PNConfiguration(); +$config->setPublishKey('demo'); +$config->setSubscribeKey('demo'); +$config->setUuid('example'); + +$pubnub = new PubNub($config); +$client = new CustomClient(); +$pubnub->setClient($client); + +$time = $pubnub->time()->sync(); +print_r($time->getTimetoken()); diff --git a/examples/simple_partial_updates.php b/examples/simple_partial_updates.php new file mode 100644 index 00000000..14c588b4 --- /dev/null +++ b/examples/simple_partial_updates.php @@ -0,0 +1,58 @@ + +setPublishKey("demo"); +$pnconf->setSubscribeKey("demo"); +$pnconf->setUuid("example"); + +$pubnub = new PubNub($pnconf); + +$channel = "demo_example"; +$channelName = "Channel1on1"; +$channelDescription = "Channel for 1on1 conversation"; +$status = "active"; +$type = "1on1"; +$initialCustom = ["Days" => "Mon-Fri"]; + +// Set initial channel metadata +$pubnub->setChannelMetadata() + ->channel($channel) + ->meta(["custom" => $initialCustom]) + ->status($status) + ->type($type) + ->sync(); + +// Fetch the current metadata +$response = $pubnub->getChannelMetadata() + ->channel($channel) + ->includeCustom(true) + ->sync(); + +$custom = (array)$response->getCustom(); +$additionalMetadata = ["Months" => "Jan-May"]; + +// Merge additional metadata +$updatedCustomMetadata = array_merge($custom, $additionalMetadata); + +// Update the channel metadata +$updatedMetadata = $pubnub->setChannelMetadata() + ->channel($channel) + ->custom($updatedCustomMetadata) + ->includeCustom(true) + ->sync(); + +print("Updated channel metadata:"); +print_r($updatedMetadata->getData()); + +// Cleanup +$pubnub->removeChannelMetadata() + ->channel($channel) + ->sync(); diff --git a/phpstan-baseline.neon b/phpstan-baseline.neon index 73dd8381..a9155b9f 100644 --- a/phpstan-baseline.neon +++ b/phpstan-baseline.neon @@ -605,7 +605,6 @@ parameters: count: 1 path: src/PubNub/Endpoints/Endpoint.php - - message: "#^Method PubNub\\\\Endpoints\\\\Endpoint\\:\\:createStatus\\(\\) has parameter \\$response with no type specified\\.$#" count: 1 @@ -656,11 +655,6 @@ parameters: count: 1 path: src/PubNub/Endpoints/Endpoint.php - - - message: "#^Method PubNub\\\\Endpoints\\\\Endpoint\\:\\:getTransport\\(\\) has no return type specified\\.$#" - count: 1 - path: src/PubNub/Endpoints/Endpoint.php - - message: "#^Method PubNub\\\\Endpoints\\\\Endpoint\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" count: 1 @@ -693,11 +687,6 @@ parameters: - message: "#^Negated boolean expression is always false\\.$#" - count: 2 - path: src/PubNub/Endpoints/Endpoint.php - - - - message: "#^Parameter \\#1 \\$result of method PubNub\\\\Endpoints\\\\Endpoint\\:\\:createResponse\\(\\) expects array, WpOrg\\\\Requests\\\\Response given\\.$#" count: 1 path: src/PubNub/Endpoints/Endpoint.php @@ -706,21 +695,6 @@ parameters: count: 1 path: src/PubNub/Endpoints/Endpoint.php - - - message: "#^Parameter \\#3 \\$data of static method WpOrg\\\\Requests\\\\Requests\\:\\:request\\(\\) expects array\\|null, string\\|null given\\.$#" - count: 1 - path: src/PubNub/Endpoints/Endpoint.php - - - - message: "#^Parameter \\#5 \\$authKey of class PubNub\\\\Models\\\\ResponseHelpers\\\\ResponseInfo constructor expects string, null given\\.$#" - count: 1 - path: src/PubNub/Endpoints/Endpoint.php - - - - message: "#^Property PubNub\\\\Endpoints\\\\Endpoint\\:\\:\\$cachedTransports type has no value type specified in iterable type array\\.$#" - count: 1 - path: src/PubNub/Endpoints/Endpoint.php - - message: "#^Property PubNub\\\\Endpoints\\\\Endpoint\\:\\:\\$customHost has no type specified\\.$#" count: 1 @@ -2996,11 +2970,6 @@ parameters: count: 1 path: src/PubNub/Models/Consumer/FileSharing/PNFetchFileUploadS3DataResult.php - - - message: "#^Method PubNub\\\\Models\\\\Consumer\\\\FileSharing\\\\PNGetFileDownloadURLResult\\:\\:__construct\\(\\) has parameter \\$result with no type specified\\.$#" - count: 1 - path: src/PubNub/Models/Consumer/FileSharing/PNGetFileDownloadURLResult.php - - message: "#^Method PubNub\\\\Models\\\\Consumer\\\\FileSharing\\\\PNGetFileDownloadURLResult\\:\\:getFileUrl\\(\\) has no return type specified\\.$#" count: 1 @@ -5256,81 +5225,11 @@ parameters: count: 1 path: tests/functional/ListChannelsInChannelGroupTest.php - - - message: "#^Method Tests\\\\Functional\\\\MessageCountExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/functional/MessageCountTest.php - - - - message: "#^Method Tests\\\\Functional\\\\MessageCountExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/functional/MessageCountTest.php - - - - message: "#^Method Tests\\\\Functional\\\\MessageCountExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/functional/MessageCountTest.php - - - - message: "#^Method Tests\\\\Functional\\\\MessageCountTest\\:\\:testChannel_withMultiEmptyToken\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/functional/MessageCountTest.php - - - - message: "#^Method Tests\\\\Functional\\\\MessageCountTest\\:\\:testChannel_withMultiNullToken\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/functional/MessageCountTest.php - - - - message: "#^Method Tests\\\\Functional\\\\MessageCountTest\\:\\:testMultiChannel_withMultiTimestamp\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/functional/MessageCountTest.php - - - - message: "#^Method Tests\\\\Functional\\\\MessageCountTest\\:\\:testMultiChannel_withSingleTimestamp\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/functional/MessageCountTest.php - - - - message: "#^Method Tests\\\\Functional\\\\MessageCountTest\\:\\:testSingleChannel_withMultiTimestamp\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/functional/MessageCountTest.php - - - - message: "#^Method Tests\\\\Functional\\\\MessageCountTest\\:\\:testSingleChannel_withSingleTimestamp\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/functional/MessageCountTest.php - - - - message: "#^Method Tests\\\\Functional\\\\MessageCountTest\\:\\:testSyncDisabled\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/functional/MessageCountTest.php - - - - message: "#^Method Tests\\\\Functional\\\\MessageCountTest\\:\\:testWithoutChannels_SingleTimeToken\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/functional/MessageCountTest.php - - - - message: "#^Method Tests\\\\Functional\\\\MessageCountTest\\:\\:testWithoutChannels_TimeTokenList\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/functional/MessageCountTest.php - - - - message: "#^Method Tests\\\\Functional\\\\MessageCountTest\\:\\:testWithoutTimeToken\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/functional/MessageCountTest.php - - message: "#^Parameter \\#1 \\$channelsTimetoken of method PubNub\\\\Endpoints\\\\MessageCount\\:\\:channelsTimetoken\\(\\) expects array, null given\\.$#" count: 1 path: tests/functional/MessageCountTest.php - - - message: "#^Property Tests\\\\Functional\\\\MessageCountExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/functional/MessageCountTest.php - - message: "#^Method Tests\\\\Functional\\\\PublishTest\\:\\:assertGeneratesCorrectPath\\(\\) has no return type specified\\.$#" count: 1 @@ -5901,21 +5800,6 @@ parameters: count: 1 path: tests/functional/push/ListPushProvisionsTest.php - - - message: "#^Method Tests\\\\Functional\\\\Push\\\\ListPushProvisionsExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/functional/push/ListPushProvisionsTest.php - - - - message: "#^Method Tests\\\\Functional\\\\Push\\\\ListPushProvisionsExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/functional/push/ListPushProvisionsTest.php - - - - message: "#^Method Tests\\\\Functional\\\\Push\\\\ListPushProvisionsExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/functional/push/ListPushProvisionsTest.php - - message: "#^Method Tests\\\\Functional\\\\Push\\\\ListPushProvisionsTest\\:\\:testListChannelGroupAPNS\\(\\) has no return type specified\\.$#" count: 1 @@ -5936,11 +5820,6 @@ parameters: count: 1 path: tests/functional/push/ListPushProvisionsTest.php - - - message: "#^Property Tests\\\\Functional\\\\Push\\\\ListPushProvisionsExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/functional/push/ListPushProvisionsTest.php - - message: "#^Method Tests\\\\Functional\\\\Push\\\\RemoveChannelsFromPushExposed\\:\\:buildParams\\(\\) return type has no value type specified in iterable type array\\.$#" count: 1 @@ -5981,116 +5860,6 @@ parameters: count: 1 path: tests/functional/push/RemoveDeviceFromPushTest.php - - - message: "#^Method Tests\\\\Helpers\\\\Stub\\:\\:isPathMatch\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/helpers/Stub.php - - - - message: "#^Method Tests\\\\Helpers\\\\Stub\\:\\:isPathMatch\\(\\) has parameter \\$path with no type specified\\.$#" - count: 1 - path: tests/helpers/Stub.php - - - - message: "#^Method Tests\\\\Helpers\\\\Stub\\:\\:isQueryMatch\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/helpers/Stub.php - - - - message: "#^Method Tests\\\\Helpers\\\\Stub\\:\\:isQueryMatch\\(\\) has parameter \\$actualQueryString with no type specified\\.$#" - count: 1 - path: tests/helpers/Stub.php - - - - message: "#^Method Tests\\\\Helpers\\\\Stub\\:\\:queryString\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/helpers/Stub.php - - - - message: "#^Method Tests\\\\Helpers\\\\Stub\\:\\:stripKeys\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/helpers/Stub.php - - - - message: "#^Method Tests\\\\Helpers\\\\Stub\\:\\:stripKeys\\(\\) has parameter \\$path with no type specified\\.$#" - count: 1 - path: tests/helpers/Stub.php - - - - message: "#^Method Tests\\\\Helpers\\\\Stub\\:\\:withQuery\\(\\) has parameter \\$query with no value type specified in iterable type array\\.$#" - count: 1 - path: tests/helpers/Stub.php - - - - message: "#^PHPDoc tag @var above a method has no effect\\.$#" - count: 1 - path: tests/helpers/Stub.php - - - - message: "#^PHPDoc tag @var does not specify variable name\\.$#" - count: 1 - path: tests/helpers/Stub.php - - - - message: "#^Property Tests\\\\Helpers\\\\Stub\\:\\:\\$host has no type specified\\.$#" - count: 1 - path: tests/helpers/Stub.php - - - - message: "#^Property Tests\\\\Helpers\\\\Stub\\:\\:\\$initialUrl has no type specified\\.$#" - count: 1 - path: tests/helpers/Stub.php - - - - message: "#^Property Tests\\\\Helpers\\\\Stub\\:\\:\\$method has no type specified\\.$#" - count: 1 - path: tests/helpers/Stub.php - - - - message: "#^Property Tests\\\\Helpers\\\\Stub\\:\\:\\$path has no type specified\\.$#" - count: 1 - path: tests/helpers/Stub.php - - - - message: "#^Property Tests\\\\Helpers\\\\Stub\\:\\:\\$scheme has no type specified\\.$#" - count: 1 - path: tests/helpers/Stub.php - - - - message: "#^Method Tests\\\\Helpers\\\\StubTransport\\:\\:request\\(\\) has parameter \\$data with no value type specified in iterable type array\\.$#" - count: 1 - path: tests/helpers/StubTransport.php - - - - message: "#^Method Tests\\\\Helpers\\\\StubTransport\\:\\:request\\(\\) has parameter \\$headers with no value type specified in iterable type array\\.$#" - count: 1 - path: tests/helpers/StubTransport.php - - - - message: "#^Method Tests\\\\Helpers\\\\StubTransport\\:\\:request\\(\\) has parameter \\$options with no value type specified in iterable type array\\.$#" - count: 1 - path: tests/helpers/StubTransport.php - - - - message: "#^Method Tests\\\\Helpers\\\\StubTransport\\:\\:request_multiple\\(\\) has parameter \\$options with no value type specified in iterable type array\\.$#" - count: 1 - path: tests/helpers/StubTransport.php - - - - message: "#^Method Tests\\\\Helpers\\\\StubTransport\\:\\:request_multiple\\(\\) has parameter \\$requests with no value type specified in iterable type array\\.$#" - count: 1 - path: tests/helpers/StubTransport.php - - - - message: "#^Method Tests\\\\Helpers\\\\StubTransport\\:\\:request_multiple\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/helpers/StubTransport.php - - - - message: "#^Property Tests\\\\Helpers\\\\StubTransport\\:\\:\\$requestsCountInt has no type specified\\.$#" - count: 1 - path: tests/helpers/StubTransport.php - - message: "#^Method Tests\\\\Integrational\\\\AddChannelChannelGroupEndpointTest\\:\\:testChannelMissing\\(\\) has no return type specified\\.$#" count: 1 @@ -6126,41 +5895,6 @@ parameters: count: 1 path: tests/integrational/AddChannelChannelGroupEndpointTest.php - - - message: "#^Method Tests\\\\Integrational\\\\AddChannelChannelGroupExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/AddChannelChannelGroupEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\AddChannelChannelGroupExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/AddChannelChannelGroupEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\AddChannelChannelGroupExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/AddChannelChannelGroupEndpointTest.php - - - - message: "#^Property Tests\\\\Integrational\\\\AddChannelChannelGroupExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/AddChannelChannelGroupEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\FetchMessagesExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/FetchMessagesTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\FetchMessagesExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/FetchMessagesTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\FetchMessagesExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/FetchMessagesTest.php - - message: "#^Method Tests\\\\Integrational\\\\FetchMessagesTest\\:\\:testFetchEncrypted\\(\\) has no return type specified\\.$#" count: 1 @@ -6181,11 +5915,6 @@ parameters: count: 1 path: tests/integrational/FetchMessagesTest.php - - - message: "#^Property Tests\\\\Integrational\\\\FetchMessagesExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/FetchMessagesTest.php - - message: "#^Property Tests\\\\Integrational\\\\FetchMessagesTest\\:\\:\\$endTimetoken has no type specified\\.$#" count: 1 @@ -6236,21 +5965,6 @@ parameters: count: 1 path: tests/integrational/FilesTest.php - - - message: "#^Method Tests\\\\Integrational\\\\GetStateExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/GetStateTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\GetStateExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/GetStateTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\GetStateExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/GetStateTest.php - - message: "#^Method Tests\\\\Integrational\\\\GetStateTest\\:\\:testCombination\\(\\) has no return type specified\\.$#" count: 1 @@ -6326,31 +6040,6 @@ parameters: count: 1 path: tests/integrational/GetStateTest.php - - - message: "#^Property Tests\\\\Integrational\\\\GetStateExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/GetStateTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\HereNowExposed\\:\\:__construct\\(\\) has parameter \\$pubnubInstance with no type specified\\.$#" - count: 1 - path: tests/integrational/HereNowTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\HereNowExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/HereNowTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\HereNowExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/HereNowTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\HereNowExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/HereNowTest.php - - message: "#^Method Tests\\\\Integrational\\\\HereNowTest\\:\\:testEmptySubKey\\(\\) has no return type specified\\.$#" count: 1 @@ -6416,46 +6105,6 @@ parameters: count: 1 path: tests/integrational/HereNowTest.php - - - message: "#^Property Tests\\\\Integrational\\\\HereNowExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/HereNowTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\HistoryDeleteExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/HistoryDeleteTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\HistoryDeleteExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/HistoryDeleteTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\HistoryDeleteExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/HistoryDeleteTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\TestPubNubHistoryDelete\\:\\:testChannelIsEmptyException\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/HistoryDeleteTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\TestPubNubHistoryDelete\\:\\:testMissingChannelException\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/HistoryDeleteTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\TestPubNubHistoryDelete\\:\\:testSuperCallTest\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/HistoryDeleteTest.php - - - - message: "#^Property Tests\\\\Integrational\\\\HistoryDeleteExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/HistoryDeleteTest.php - - message: "#^Access to offset 'a' on an unknown class PubNub\\\\Models\\\\Consumer\\\\History\\\\any\\.$#" count: 6 @@ -6477,32 +6126,17 @@ parameters: path: tests/integrational/HistoryTest.php - - message: "#^Method Tests\\\\Integrational\\\\HistoryExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" + message: "#^Method Tests\\\\Integrational\\\\HistoryTest\\:\\:testAuthSuccess\\(\\) has no return type specified\\.$#" count: 1 path: tests/integrational/HistoryTest.php - - message: "#^Method Tests\\\\Integrational\\\\HistoryExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" + message: "#^Method Tests\\\\Integrational\\\\HistoryTest\\:\\:testChannelIsEmpty\\(\\) has no return type specified\\.$#" count: 1 path: tests/integrational/HistoryTest.php - - message: "#^Method Tests\\\\Integrational\\\\HistoryExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/HistoryTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\HistoryTest\\:\\:testAuthSuccess\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/HistoryTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\HistoryTest\\:\\:testChannelIsEmpty\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/HistoryTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\HistoryTest\\:\\:testCountReverseStartEndSuccess\\(\\) has no return type specified\\.$#" + message: "#^Method Tests\\\\Integrational\\\\HistoryTest\\:\\:testCountReverseStartEndSuccess\\(\\) has no return type specified\\.$#" count: 1 path: tests/integrational/HistoryTest.php @@ -6551,31 +6185,11 @@ parameters: count: 1 path: tests/integrational/HistoryTest.php - - - message: "#^Property Tests\\\\Integrational\\\\HistoryExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/HistoryTest.php - - message: "#^Unreachable statement \\- code above always terminates\\.$#" count: 1 path: tests/integrational/HistoryTest.php - - - message: "#^Method Tests\\\\Integrational\\\\ListChannelsInChannelGroupExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/ListChannelsInChannelGroupTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\ListChannelsInChannelGroupExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/ListChannelsInChannelGroupTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\ListChannelsInChannelGroupExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/ListChannelsInChannelGroupTest.php - - message: "#^Method Tests\\\\Integrational\\\\ListChannelsInChannelGroupTest\\:\\:testEmptyGroup\\(\\) has no return type specified\\.$#" count: 1 @@ -6606,36 +6220,6 @@ parameters: count: 1 path: tests/integrational/ListChannelsInChannelGroupTest.php - - - message: "#^Property Tests\\\\Integrational\\\\ListChannelsInChannelGroupExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/ListChannelsInChannelGroupTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\ListPushProvisionsExposed\\:\\:buildParams\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/ListPushProvisionsTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\ListPushProvisionsExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/ListPushProvisionsTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\ListPushProvisionsExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/ListPushProvisionsTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\ListPushProvisionsExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/ListPushProvisionsTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\ListPushProvisionsTest\\:\\:superCallTest\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/ListPushProvisionsTest.php - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\ListPushProvisionsTest\\:\\:testAppleSuccess\\(\\) has no return type specified\\.$#" count: 1 @@ -6686,31 +6270,6 @@ parameters: count: 1 path: tests/integrational/ListPushProvisionsTest.php - - - message: "#^Property Tests\\\\Integrational\\\\Push\\\\ListPushProvisionsExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/ListPushProvisionsTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\AddChannelsToPushExposed\\:\\:buildParams\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/ModifyPushChannelsForDeviceTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\AddChannelsToPushExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/ModifyPushChannelsForDeviceTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\AddChannelsToPushExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/ModifyPushChannelsForDeviceTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\AddChannelsToPushExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/ModifyPushChannelsForDeviceTest.php - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\ModifyPushChannelsForDeviceTest\\:\\:superCallTest\\(\\) has no return type specified\\.$#" count: 1 @@ -6861,66 +6420,11 @@ parameters: count: 1 path: tests/integrational/ModifyPushChannelsForDeviceTest.php - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\RemoveChannelsFromPushTestExposed\\:\\:buildParams\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/ModifyPushChannelsForDeviceTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\RemoveChannelsFromPushTestExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/ModifyPushChannelsForDeviceTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\RemoveChannelsFromPushTestExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/ModifyPushChannelsForDeviceTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\RemoveChannelsFromPushTestExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/ModifyPushChannelsForDeviceTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\RemovePushNotificationsFromChannelsExposed\\:\\:buildParams\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/ModifyPushChannelsForDeviceTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\RemovePushNotificationsFromChannelsExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/ModifyPushChannelsForDeviceTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\RemovePushNotificationsFromChannelsExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/ModifyPushChannelsForDeviceTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\RemovePushNotificationsFromChannelsExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/ModifyPushChannelsForDeviceTest.php - - message: "#^Parameter \\#1 \\$subscribeKey of method PubNub\\\\PNConfiguration\\:\\:setSubscribeKey\\(\\) expects string, null given\\.$#" count: 3 path: tests/integrational/ModifyPushChannelsForDeviceTest.php - - - message: "#^Property Tests\\\\Integrational\\\\Push\\\\AddChannelsToPushExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/ModifyPushChannelsForDeviceTest.php - - - - message: "#^Property Tests\\\\Integrational\\\\Push\\\\RemoveChannelsFromPushTestExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/ModifyPushChannelsForDeviceTest.php - - - - message: "#^Property Tests\\\\Integrational\\\\Push\\\\RemovePushNotificationsFromChannelsExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/ModifyPushChannelsForDeviceTest.php - - message: "#^Method Tests\\\\Integrational\\\\PamTest\\:\\:testGlobalLevel\\(\\) has no return type specified\\.$#" count: 1 @@ -7086,26 +6590,6 @@ parameters: count: 1 path: tests/integrational/RemoveChannelFromChannelGroupEndpointTest.php - - - message: "#^Method Tests\\\\Integrational\\\\RemoveChannelFromChannelGroupExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/RemoveChannelFromChannelGroupEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\RemoveChannelFromChannelGroupExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/RemoveChannelFromChannelGroupEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\RemoveChannelFromChannelGroupExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/RemoveChannelFromChannelGroupEndpointTest.php - - - - message: "#^Property Tests\\\\Integrational\\\\RemoveChannelFromChannelGroupExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/RemoveChannelFromChannelGroupEndpointTest.php - - message: "#^Method Tests\\\\Integrational\\\\RemoveChannelGroupEndpointTest\\:\\:superCallTest\\(\\) has no return type specified\\.$#" count: 1 @@ -7131,46 +6615,6 @@ parameters: count: 1 path: tests/integrational/RemoveChannelGroupEndpointTest.php - - - message: "#^Method Tests\\\\Integrational\\\\RemoveChannelGroupExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/RemoveChannelGroupEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\RemoveChannelGroupExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/RemoveChannelGroupEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\RemoveChannelGroupExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/RemoveChannelGroupEndpointTest.php - - - - message: "#^Property Tests\\\\Integrational\\\\RemoveChannelGroupExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/RemoveChannelGroupEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\SetStateExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/SetStateTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\SetStateExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/SetStateTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\SetStateExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/SetStateTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\SetStateTest\\:\\:testApplyNon200\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/SetStateTest.php - - message: "#^Method Tests\\\\Integrational\\\\SetStateTest\\:\\:testApplyStateForChannel\\(\\) has no return type specified\\.$#" count: 1 @@ -7232,60 +6676,15 @@ parameters: path: tests/integrational/SetStateTest.php - - message: "#^Parameter \\#1 \\$subscribeKey of method PubNub\\\\PNConfiguration\\:\\:setSubscribeKey\\(\\) expects string, null given\\.$#" + message: "#^Parameter \\#1 \\$state of method PubNub\\\\Endpoints\\\\Presence\\\\SetState\\:\\:state\\(\\) expects array, null given\\.$#" count: 1 path: tests/integrational/SetStateTest.php - - message: "#^Property Tests\\\\Integrational\\\\SetStateExposed\\:\\:\\$transport has no type specified\\.$#" + message: "#^Parameter \\#1 \\$subscribeKey of method PubNub\\\\PNConfiguration\\:\\:setSubscribeKey\\(\\) expects string, null given\\.$#" count: 1 path: tests/integrational/SetStateTest.php - - - message: "#^Method Tests\\\\Integrational\\\\CheckSslTransport\\:\\:isRequestedSecureOrigin\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/SslTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\CheckSslTransport\\:\\:request\\(\\) has parameter \\$data with no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/SslTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\CheckSslTransport\\:\\:request\\(\\) has parameter \\$headers with no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/SslTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\CheckSslTransport\\:\\:request\\(\\) has parameter \\$options with no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/SslTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\CheckSslTransport\\:\\:request_multiple\\(\\) has parameter \\$options with no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/SslTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\CheckSslTransport\\:\\:request_multiple\\(\\) has parameter \\$requests with no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/SslTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\CheckSslTransport\\:\\:request_multiple\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/SslTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\CheckSslTransport\\:\\:request_multiple\\(\\) should return array but return statement is missing\\.$#" - count: 1 - path: tests/integrational/SslTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\CheckSslTransport\\:\\:test\\(\\) should return bool but return statement is missing\\.$#" - count: 1 - path: tests/integrational/SslTest.php - - message: "#^Method Tests\\\\Integrational\\\\SslTest\\:\\:testSslCanBeDisabled\\(\\) has no return type specified\\.$#" count: 1 @@ -7296,11 +6695,6 @@ parameters: count: 1 path: tests/integrational/SslTest.php - - - message: "#^Property Tests\\\\Integrational\\\\CheckSslTransport\\:\\:\\$requestedThroughHttps has no type specified\\.$#" - count: 1 - path: tests/integrational/SslTest.php - - message: "#^Else branch is unreachable because previous condition is always true\\.$#" count: 1 @@ -7646,26 +7040,6 @@ parameters: count: 1 path: tests/integrational/WhereNowTest.php - - - message: "#^Method Tests\\\\Integrational\\\\WhereNowTestExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/WhereNowTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\WhereNowTestExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/WhereNowTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\WhereNowTestExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/WhereNowTest.php - - - - message: "#^Property Tests\\\\Integrational\\\\WhereNowTestExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/WhereNowTest.php - - message: "#^Method Tests\\\\Integrational\\\\Objects\\\\Channel\\\\GetAllChannelMetadataEndpointTest\\:\\:testGetAllChannelMetadata\\(\\) has no return type specified\\.$#" count: 1 @@ -7736,21 +7110,6 @@ parameters: count: 1 path: tests/integrational/objects/uuid/SetUUIDMetadataEndpointTest.php - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\AddChannelsToPushEndpointExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/push/AddChannelsToPushEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\AddChannelsToPushEndpointExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/push/AddChannelsToPushEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\AddChannelsToPushEndpointExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/push/AddChannelsToPushEndpointTest.php - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\AddChannelsToPushEndpointTest\\:\\:testPushAddApns2\\(\\) has no return type specified\\.$#" count: 1 @@ -7776,31 +7135,6 @@ parameters: count: 1 path: tests/integrational/push/AddChannelsToPushEndpointTest.php - - - message: "#^Property Tests\\\\Integrational\\\\Push\\\\AddChannelsToPushEndpointExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/push/AddChannelsToPushEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\ListPushProvisionsEndpointExposed\\:\\:buildParams\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/push/ListPushProvisionsEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\ListPushProvisionsEndpointExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/push/ListPushProvisionsEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\ListPushProvisionsEndpointExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/push/ListPushProvisionsEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\ListPushProvisionsEndpointExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/push/ListPushProvisionsEndpointTest.php - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\ListPushProvisionsEndpointTest\\:\\:testListChannelGroupAPNS\\(\\) has no return type specified\\.$#" count: 1 @@ -7826,26 +7160,6 @@ parameters: count: 1 path: tests/integrational/push/ListPushProvisionsEndpointTest.php - - - message: "#^Property Tests\\\\Integrational\\\\Push\\\\ListPushProvisionsEndpointExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/push/ListPushProvisionsEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\RemoveChannelsFromPushEndpointExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/push/RemoveChannelsFromPushEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\RemoveChannelsFromPushEndpointExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/push/RemoveChannelsFromPushEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\RemoveChannelsFromPushEndpointExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/push/RemoveChannelsFromPushEndpointTest.php - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\RemoveChannelsFromPushEndpointTest\\:\\:testPushRemoveApns2\\(\\) has no return type specified\\.$#" count: 1 @@ -7866,26 +7180,6 @@ parameters: count: 1 path: tests/integrational/push/RemoveChannelsFromPushEndpointTest.php - - - message: "#^Property Tests\\\\Integrational\\\\Push\\\\RemoveChannelsFromPushEndpointExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/push/RemoveChannelsFromPushEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\RemoveDeviceFromPushEndpointExposed\\:\\:requestOptions\\(\\) return type has no value type specified in iterable type array\\.$#" - count: 1 - path: tests/integrational/push/RemoveDeviceFromPushEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\RemoveDeviceFromPushEndpointExposed\\:\\:stubFor\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/integrational/push/RemoveDeviceFromPushEndpointTest.php - - - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\RemoveDeviceFromPushEndpointExposed\\:\\:stubFor\\(\\) has parameter \\$url with no type specified\\.$#" - count: 1 - path: tests/integrational/push/RemoveDeviceFromPushEndpointTest.php - - message: "#^Method Tests\\\\Integrational\\\\Push\\\\RemoveDeviceFromPushEndpointTest\\:\\:testRemovePushAPNS\\(\\) has no return type specified\\.$#" count: 1 @@ -7911,11 +7205,6 @@ parameters: count: 1 path: tests/integrational/push/RemoveDeviceFromPushEndpointTest.php - - - message: "#^Property Tests\\\\Integrational\\\\Push\\\\RemoveDeviceFromPushEndpointExposed\\:\\:\\$transport has no type specified\\.$#" - count: 1 - path: tests/integrational/push/RemoveDeviceFromPushEndpointTest.php - - message: "#^PHPDoc tag @param for parameter \\$expected with type PubNub\\\\Crypto\\\\Header is incompatible with native type string\\.$#" count: 1 @@ -8006,26 +7295,6 @@ parameters: count: 1 path: tests/unit/PublishTest.php - - - message: "#^Method StubTest\\:\\:testAny\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/unit/StubTest.php - - - - message: "#^Method StubTest\\:\\:testExtraActualArgument\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/unit/StubTest.php - - - - message: "#^Method StubTest\\:\\:testExtraExpectedArgument\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/unit/StubTest.php - - - - message: "#^Method StubTest\\:\\:testSimpleStub\\(\\) has no return type specified\\.$#" - count: 1 - path: tests/unit/StubTest.php - - message: "#^Method TelemetryManagerTest\\:\\:testAverageLatency\\(\\) has no return type specified\\.$#" count: 1 diff --git a/src/PubNub/Endpoints/Endpoint.php b/src/PubNub/Endpoints/Endpoint.php index d73907a6..2179febc 100755 --- a/src/PubNub/Endpoints/Endpoint.php +++ b/src/PubNub/Endpoints/Endpoint.php @@ -16,13 +16,10 @@ use PubNub\Models\ResponseHelpers\ResponseInfo; use PubNub\PubNub; use PubNub\PubNubUtil; -use WpOrg\Requests\Requests; -use WpOrg\Requests\Exception as RequestsException; -use WpOrg\Requests\Exception\Transport\Curl as RequestsTransportCurlException; -use WpOrg\Requests\Exception\Http\StatusUnknown as ReuestsHttpStatusUnknownException; -use WpOrg\Requests\Exception\Http as RequestsHttpException; -use WpOrg\Requests\Transport\Curl; -use WpOrg\Requests\Transport\Fsockopen; +use Psr\Http\Client\NetworkExceptionInterface; +use Psr\Http\Client\ClientExceptionInterface; +use Psr\Http\Message\RequestInterface; +use Psr\Http\Message\ResponseInterface; abstract class Endpoint { @@ -46,9 +43,6 @@ abstract class Endpoint protected $customHost = null; - /** @var array */ - protected static $cachedTransports = []; - protected $followRedirects = true; public function __construct(PubNub $pubnubInstance) @@ -59,7 +53,7 @@ public function __construct(PubNub $pubnubInstance) abstract protected function validateParams(); /** - * @param array $result Decoded json + * @param array $result * @return mixed */ abstract protected function createResponse($result); @@ -341,198 +335,130 @@ public function clear() */ protected function invokeRequestAndCacheIt() { + if ($this->envelope === null) { - $this->envelope = $this->invokeRequest(); + $this->envelope = $this->sendRequest($this->getRequest()); } return $this->envelope; } - /** - * @return array - */ - protected function requestOptions() - { - return [ - 'timeout' => $this->getRequestTimeout(), - 'connect_timeout' => $this->getConnectTimeout(), - 'transport' => $this->getTransport(), - 'useragent' => 'PHP/' . PHP_VERSION, - 'follow_redirects' => $this->followRedirects, - ]; - } - - protected function getTransport() - { - return $this->pubnub->getConfiguration()->getTransport() ?? $this->getDefaultTransport(); - } - /** * @return PNEnvelope */ - protected function invokeRequest() + protected function sendRequest(RequestInterface $request): PNEnvelope { - $headers = array_merge($this->defaultHeaders(), $this->customHeaders()); - - $url = PubNubUtil::buildUrl( - $this->pubnub->getBasePath($this->customHost), - $this->buildPath(), - $this->buildParams() - ); - $data = $this->buildData(); - $method = $this->httpMethod(); - $options = $this->requestOptions(); - - $this->pubnub->getLogger()->debug($method . " " . $url, ['method' => $this->getName()]); - - if ($data) { - $this->pubnub->getLogger()->debug("Body:\n" . $data, ['method' => $this->getName()]); - } - - $statusCategory = PNStatusCategory::PNUnknownCategory; - $requestTimeStart = microtime(true); + $client = $this->pubnub->getClient(); try { - $request = Requests::request($url, $headers, $data, $method, $options); - } catch (ReuestsHttpStatusUnknownException $e) { - $this->pubnub->getLogger()->error($e->getMessage(), ['method' => $this->getName()]); - - return new PNEnvelope($e->getData(), $this->createStatus( - $statusCategory, + if (is_callable([$client, 'send'])) { + $response = $client->send($request, $this->requestOptions()); + } else { + $response = $client->sendRequest($request); + } + $envelope = $this->parseResponse($response); + } catch (NetworkExceptionInterface $exception) { + return new PNEnvelope(null, $this->createStatus( + PNStatusCategory::PNTimeoutCategory, null, null, - (new PubNubConnectionException())->setOriginalException($e) + (new PubNubConnectionException())->setOriginalException($exception) )); - } catch (RequestsTransportCurlException $e) { - $this->pubnub->getLogger()->error($e->getMessage(), ['method' => $this->getName()]); + } catch (ClientExceptionInterface $exception) { + $statusCode = $exception->getCode(); + $response = substr($exception->getMessage(), strpos($exception->getMessage(), "\n") + 1); + $pnServerException = new PubNubServerException(); + $pnServerException->setRawBody($response); + $pnServerException->setStatusCode($exception->getCode()); - return new PNEnvelope($e->getData(), $this->createStatus( - $statusCategory, - null, - null, - (new PubNubConnectionException())->setOriginalException($e) - )); - } catch (RequestsHttpException $e) { - $this->pubnub->getLogger()->error($e->getMessage(), ['method' => $this->getName()]); + $uuid = null; + $authKey = null; - return new PNEnvelope($e->getData(), $this->createStatus( - $statusCategory, - null, - null, - (new PubNubConnectionException())->setOriginalException($e) - )); - } catch (RequestsException $e) { - if ($e->getType() === 'curlerror' && strpos($e->getMessage(), "cURL error 28") === 0) { - $statusCategory = PNStatusCategory::PNTimeoutCategory; - } + parse_str($request->getUri()->getQuery(), $query); - $this->pubnub->getLogger()->error($e->getMessage(), ['method' => $this->getName()]); + if (array_key_exists('uuid', $query) && strlen($query['uuid']) > 0) { + $uuid = $query['uuid']; + } - return new PNEnvelope(null, $this->createStatus( - $statusCategory, - null, - null, - (new PubNubConnectionException())->setOriginalException($e) - )); - } catch (\Exception $e) { - $this->pubnub->getLogger()->error($e->getMessage(), ['method' => $this->getName()]); + if (array_key_exists('auth', $query) && strlen($query['auth']) > 0) { + $authKey = $query['auth']; + } - return new PNEnvelope(null, $this->createStatus( - $statusCategory, - null, - null, - (new PubNubConnectionException())->setOriginalException($e) - )); + $responseInfo = new ResponseInfo( + $statusCode, + $request->getUri()->getScheme() === 'https', + $request->getUri()->getHost(), + $uuid, + $authKey, + null + ); + $statusCategory = PNStatusCategory::PNBadRequestCategory; + return new PNEnvelope(null, $this->createStatus($statusCategory, null, $responseInfo, $pnServerException)); } + return $envelope; + } - $url = parse_url($url); - $query = []; + /** + * @param ResponseInterface $response + * @return PNEnvelope + */ + public function parseResponse(ResponseInterface $response): PNEnvelope + { + $result = null; + $status = null; - if (array_key_exists('query', $url)) { - parse_str($url['query'], $query); - } + $statusCode = $response->getStatusCode(); + $statusCategory = PNStatusCategory::PNUnknownCategory; + + $request = $this->getRequest(); $uuid = null; $authKey = null; + parse_str($request->getUri()->getQuery(), $query); if (array_key_exists('uuid', $query) && strlen($query['uuid']) > 0) { $uuid = $query['uuid']; } if (array_key_exists('auth', $query) && strlen($query['auth']) > 0) { - $uuid = $query['auth']; + $authKey = $query['auth']; } - $responseInfo = new ResponseInfo( - $request->status_code, - $url['scheme'] == 'https', - $url['host'], + $statusCode, + $request->getUri()->getScheme() === 'https', + $request->getUri()->getHost(), $uuid, $authKey, - $request + $response ); - if ($request->status_code == 200) { - $requestTimeEnd = microtime(true); - - if (!!$this->pubnub->getTelemetryManager()) { - $this->pubnub->getTelemetryManager()->cleanUpTelemetryData(); - $this->pubnub->getTelemetryManager()->storeLatency( - $requestTimeEnd - $requestTimeStart, - $this->getOperationType() - ); - } - - $this->pubnub->getLogger()->debug( - "Response body: " . $request->body, - ['method' => $this->getName(), 'statusCode' => $request->status_code] - ); - + if ($statusCode === 200) { + $contents = $response->getBody()->getContents(); if (static::RESPONSE_IS_JSON) { - // NOTICE: 1 == JSON_OBJECT_AS_ARRAY (hhvm doesn't support this constant) - $parsedJSON = json_decode($request->body, true, 512, 1); - $errorMessage = json_last_error_msg(); + $parsedJSON = json_decode($contents, true); if (json_last_error()) { - $this->pubnub->getLogger()->error( - "Unable to decode JSON body: " . $request->body, - ['method' => $this->getName()] - ); - return new PNEnvelope(null, $this->createStatus( $statusCategory, - $request->body, + $response->getBody()->getContents(), $responseInfo, (new PubNubResponseParsingException()) - ->setResponseString($request->body) - ->setDescription($errorMessage) + ->setResponseString($request->getBody()) + ->setDescription(json_last_error_msg()) )); } - - return new PNEnvelope( - $this->createResponse($parsedJSON), - $this->createStatus($statusCategory, $request->body, $responseInfo, null) - ); + $result = $this->createResponse($parsedJSON); } else { - return new PNEnvelope( - $this->createResponse($request->body), - $this->createStatus($statusCategory, $request->body, $responseInfo, null) - ); + $result = $this->createResponse($contents); } - } elseif ($request->status_code === 307 && !$this->followRedirects) { - return new PNEnvelope( - $this->createResponse($request), - $this->createStatus($statusCategory, $request->body, $responseInfo, null) - ); + $status = $this->createStatus($statusCategory, $response->getBody(), $responseInfo, null); + } elseif ($statusCode === 307 && !$this->followRedirects) { + $result = $this->createResponse($response->getHeaders()); + $status = $this->createStatus($statusCategory, $response->getBody(), $responseInfo, null); } else { - $this->pubnub->getLogger()->warning( - "Response error: " . $request->body, - ['method' => $this->getName(), - 'statusCode' => $request->status_code] - ); - - switch ($request->status_code) { + $result = null; + switch ($statusCode) { case 400: $statusCategory = PNStatusCategory::PNBadRequestCategory; break; @@ -542,19 +468,52 @@ protected function invokeRequest() } $exception = (new PubNubServerException()) - ->setStatusCode($request->status_code) - ->setRawBody($request->body); + ->setStatusCode($statusCode) + ->setRawBody($response->getBody()); - // NOTICE: 530 code is for testing purposes only - if ($request->status_code === 530) { - $exception->forceMessage($request->body); - } + $status = $this->createStatus($statusCategory, $response->getBody(), $responseInfo, $exception); + } - return new PNEnvelope( - null, - $this->createStatus($statusCategory, $request->body, $responseInfo, $exception) - ); + return new PNEnvelope($result, $status); + } + + /** + * @return array + */ + protected function requestOptions() + { + return [ + 'timeout' => $this->getRequestTimeout(), + 'connect_timeout' => $this->getConnectTimeout(), + 'useragent' => 'PHP/' . PHP_VERSION, + 'allow_redirects' => (bool)$this->followRedirects, + ]; + } + + public function getRequest(): RequestInterface + { + $factory = $this->pubnub->getRequestFactory(); + $method = $this->httpMethod(); + $url = PubNubUtil::buildUrl( + $this->pubnub->getBasePath($this->customHost), + $this->buildPath(), + $this->buildParams() + ); + + $data = $this->buildData(); + $method = $this->httpMethod(); + $headers = array_merge($this->defaultHeaders(), $this->customHeaders()); + + $request = $factory->createRequest($method, $url); + + foreach ($headers as $key => $value) { + $request = $request->withHeader($key, $value); } + + if ($data) { + $request = $request->withBody(\GuzzleHttp\Psr7\Utils::streamFor($data)); + } + return $request; } /** @@ -608,47 +567,6 @@ protected function getAffectedUsers() return null; } - /** - * @return \WpOrg\Requests\Transport - * @throws \Exception - */ - private function getDefaultTransport() - { - $need_ssl = (0 === stripos($this->pubnub->getBasePath($this->customHost), 'https://')); - $capabilities = array('ssl' => $need_ssl); - - $cap_string = serialize($capabilities); - $method = $this->httpMethod(); - - if (!isset(self::$cachedTransports[$method])) { - self::$cachedTransports[$method] = []; - } - - if (isset(self::$cachedTransports[$method][$cap_string])) { - return self::$cachedTransports[$method][$cap_string]; - } - - $transports = array(Curl::class, Fsockopen::class); - - foreach ($transports as $class) { - if (!class_exists($class)) { - continue; - } - - $result = call_user_func(array($class, 'test'), $capabilities); - if ($result) { - self::$cachedTransports[$method][$cap_string] = new $class(); - break; - } - } - - if (self::$cachedTransports[$method][$cap_string] === null) { - throw new \Exception('No working transports found'); - } - - return self::$cachedTransports[$method][$cap_string]; - } - /** * @param $json * @return array diff --git a/src/PubNub/Endpoints/Objects/Channel/SetChannelMetadata.php b/src/PubNub/Endpoints/Objects/Channel/SetChannelMetadata.php index 179d8339..ad27625b 100644 --- a/src/PubNub/Endpoints/Objects/Channel/SetChannelMetadata.php +++ b/src/PubNub/Endpoints/Objects/Channel/SetChannelMetadata.php @@ -34,6 +34,8 @@ public function channel($ch) } /** + * @deprecated use setName, setDescription and setCustom instead + * * @param array $meta * @return $this */ @@ -44,6 +46,36 @@ public function meta($meta) return $this; } + /** + * @param string $name + * @return $this + */ + public function setName($name): self + { + $this->meta['name'] = $name; + return $this; + } + + /** + * @param string $description + * @return $this + */ + public function setDescription($description): self + { + $this->meta['description'] = $description; + return $this; + } + + /** + * @param string[] | string $custom + * @return $this + */ + public function setCustom($custom): self + { + $this->meta['custom'] = $custom; + return $this; + } + /** * @throws PubNubValidationException */ diff --git a/src/PubNub/Exceptions/PubNubConnectionException.php b/src/PubNub/Exceptions/PubNubConnectionException.php index 88ed0e18..b7249eb2 100644 --- a/src/PubNub/Exceptions/PubNubConnectionException.php +++ b/src/PubNub/Exceptions/PubNubConnectionException.php @@ -2,14 +2,13 @@ namespace PubNub\Exceptions; - class PubNubConnectionException extends PubNubException { - /** @var \Exception */ + /** @var \Throwable */ protected $originalException; /** - * @return \Exception + * @return \Throwable */ public function getOriginalException() { @@ -17,7 +16,7 @@ public function getOriginalException() } /** - * @param \Exception $originalException + * @param \Throwable $originalException * @return $this */ public function setOriginalException($originalException) diff --git a/src/PubNub/Models/Consumer/FileSharing/PNGetFileDownloadURLResult.php b/src/PubNub/Models/Consumer/FileSharing/PNGetFileDownloadURLResult.php index 54b0fe63..15e55a4a 100644 --- a/src/PubNub/Models/Consumer/FileSharing/PNGetFileDownloadURLResult.php +++ b/src/PubNub/Models/Consumer/FileSharing/PNGetFileDownloadURLResult.php @@ -2,13 +2,26 @@ namespace PubNub\Models\Consumer\FileSharing; +use PubNub\Exceptions\PubNubResponseParsingException; + class PNGetFileDownloadURLResult { protected string $fileUrl; - public function __construct($result) + /** + * + * @param string[] $response + * @return void + */ + public function __construct(array $response) { - $this->fileUrl = $result->headers->getAll()['location'][0]; + try { + $this->fileUrl = $response['Location'][0]; + assert(is_string($this->fileUrl)); + assert(!empty($this->fileUrl)); + } catch (\Exception) { + throw new PubNubResponseParsingException("Failed to parse response: " . json_encode($response)); + } } public function __toString() diff --git a/src/PubNub/Models/Consumer/Objects/Member/PNMemberIncludes.php b/src/PubNub/Models/Consumer/Objects/Member/PNMemberIncludes.php index 62763524..f11dcc38 100644 --- a/src/PubNub/Models/Consumer/Objects/Member/PNMemberIncludes.php +++ b/src/PubNub/Models/Consumer/Objects/Member/PNMemberIncludes.php @@ -7,7 +7,6 @@ class PNMemberIncludes extends PNIncludes { public bool $user = false; - public bool $userId = false; public bool $userCustom = false; public bool $userType = false; public bool $userStatus = false; @@ -15,8 +14,7 @@ class PNMemberIncludes extends PNIncludes public function __construct() { $this->mapping = array_merge($this->mapping, [ - 'user' => 'uuid', - 'userId' => 'uuid.id', + 'user' => 'uuid.id', 'userCustom' => 'uuid.custom', 'userType' => 'uuid.type', 'userStatus' => 'uuid.status', @@ -29,12 +27,6 @@ public function user(bool $user = true): self return $this; } - public function userId(bool $userId = true): self - { - $this->userId = $userId; - return $this; - } - public function userCustom(bool $userCustom = true): self { $this->userCustom = $userCustom; diff --git a/src/PubNub/Models/Consumer/Objects/Membership/PNMembershipIncludes.php b/src/PubNub/Models/Consumer/Objects/Membership/PNMembershipIncludes.php index 571bff08..199be38f 100644 --- a/src/PubNub/Models/Consumer/Objects/Membership/PNMembershipIncludes.php +++ b/src/PubNub/Models/Consumer/Objects/Membership/PNMembershipIncludes.php @@ -7,7 +7,6 @@ class PNMembershipIncludes extends PNIncludes { public bool $channel = false; - public bool $channelId = false; public bool $channelCustom = false; public bool $channelType = false; public bool $channelStatus = false; @@ -15,8 +14,7 @@ class PNMembershipIncludes extends PNIncludes public function __construct() { $this->mapping = array_merge($this->mapping, [ - 'channel' => 'channel', - 'channelId' => 'channel.id', + 'channel' => 'channel.id', 'channelCustom' => 'channel.custom', 'channelType' => 'channel.type', 'channelStatus' => 'channel.status', @@ -29,12 +27,6 @@ public function channel(bool $channel = true): self return $this; } - public function channelId(bool $channelId = true): self - { - $this->channelId = $channelId; - return $this; - } - public function channelCustom(bool $channelCustom = true): self { $this->channelCustom = $channelCustom; diff --git a/src/PubNub/Models/Consumer/PNTimeResult.php b/src/PubNub/Models/Consumer/PNTimeResult.php index 1b4713bb..922cd579 100755 --- a/src/PubNub/Models/Consumer/PNTimeResult.php +++ b/src/PubNub/Models/Consumer/PNTimeResult.php @@ -2,7 +2,6 @@ namespace PubNub\Models\Consumer; - class PNTimeResult { /** @var int */ @@ -25,4 +24,4 @@ public function getTimetoken() { return $this->timetoken; } -} \ No newline at end of file +} diff --git a/src/PubNub/Models/ResponseHelpers/PNEnvelope.php b/src/PubNub/Models/ResponseHelpers/PNEnvelope.php index 33cd11e1..6eec15f8 100644 --- a/src/PubNub/Models/ResponseHelpers/PNEnvelope.php +++ b/src/PubNub/Models/ResponseHelpers/PNEnvelope.php @@ -2,7 +2,6 @@ namespace PubNub\Models\ResponseHelpers; - class PNEnvelope { private $status; diff --git a/src/PubNub/Models/ResponseHelpers/PNStatus.php b/src/PubNub/Models/ResponseHelpers/PNStatus.php index 0917b890..70ed8029 100755 --- a/src/PubNub/Models/ResponseHelpers/PNStatus.php +++ b/src/PubNub/Models/ResponseHelpers/PNStatus.php @@ -2,7 +2,6 @@ namespace PubNub\Models\ResponseHelpers; - use PubNub\Enums\PNOperationType; use PubNub\Enums\PNStatusCategory; use PubNub\Exceptions\PubNubException; @@ -281,4 +280,4 @@ public function setAffectedUsers($affectedUsers) return $this; } -} \ No newline at end of file +} diff --git a/src/PubNub/Models/ResponseHelpers/ResponseInfo.php b/src/PubNub/Models/ResponseHelpers/ResponseInfo.php index 28eada1e..dcd5302b 100644 --- a/src/PubNub/Models/ResponseHelpers/ResponseInfo.php +++ b/src/PubNub/Models/ResponseHelpers/ResponseInfo.php @@ -3,6 +3,7 @@ namespace PubNub\Models\ResponseHelpers; use WpOrg\Requests\Response; +use Psr\Http\Message\ResponseInterface; class ResponseInfo { @@ -21,7 +22,7 @@ class ResponseInfo /** @var string */ private $authKey; - /** @var \Requests_Response */ + /** @var ?ResponseInterface */ private $originalResponse; /** @@ -32,10 +33,17 @@ class ResponseInfo * @param string $origin * @param string $uuid * @param string $authKey - * @param \Requests_Response $originalResponse + * @param ?ResponseInterface $originalResponse + * */ - public function __construct($statusCode, $tlsEnabled, $origin, $uuid, $authKey, Response $originalResponse) - { + public function __construct( + $statusCode, + $tlsEnabled, + $origin, + $uuid, + $authKey, + ?ResponseInterface $originalResponse + ) { $this->statusCode = $statusCode; $this->tlsEnabled = $tlsEnabled; $this->origin = $origin; @@ -85,10 +93,10 @@ public function getAuthKey() } /** - * @return \Requests_Response + * @return ResponseInterface */ public function getOriginalResponse() { return $this->originalResponse; } -} \ No newline at end of file +} diff --git a/src/PubNub/PNConfiguration.php b/src/PubNub/PNConfiguration.php index b50ab548..cdb18dc2 100755 --- a/src/PubNub/PNConfiguration.php +++ b/src/PubNub/PNConfiguration.php @@ -4,7 +4,6 @@ use PubNub\Exceptions\PubNubConfigurationException; use PubNub\Exceptions\PubNubValidationException; -use WpOrg\Requests\Transport; use PubNub\CryptoModule; class PNConfiguration @@ -62,9 +61,6 @@ class PNConfiguration /** @var int How long to keep the subscribe request running before disconnect. The value is in seconds.*/ protected int $subscribeTimeout; - /** @var Transport Custom transport implementation. */ - protected ?Transport $transport = null; - /** * @var bool * When true the initialization vector (IV) is random for all requests (not just for file upload). @@ -443,26 +439,6 @@ public function setSubscribeTimeout(int $subscribeTimeout): self return $this; } - /** - * @return Transport - */ - public function getTransport(): Transport | null - { - return $this->transport; - } - - /** - * @param Transport $transport - * @return $this - */ - public function setTransport($transport) - { - $this->checkLock(); - $this->transport = $transport; - - return $this; - } - /** * @return bool */ diff --git a/src/PubNub/PubNub.php b/src/PubNub/PubNub.php index e05d02f1..a2f118d8 100644 --- a/src/PubNub/PubNub.php +++ b/src/PubNub/PubNub.php @@ -59,10 +59,16 @@ use Psr\Log\NullLogger; use PubNub\Endpoints\FileSharing\{SendFile, DeleteFile, DownloadFile, GetFileDownloadUrl, ListFiles}; use PubNub\Models\Consumer\AccessManager\PNAccessManagerTokenResult; +use Psr\Http\Message\RequestFactoryInterface; +use Psr\Http\Client\ClientInterface; +use GuzzleHttp\Client as GuzzleHttpClient; +use GuzzleHttp\Psr7\HttpFactory as GuzzleRequestFactory; +use GuzzleHttp\Psr7\Utils as GuzzleUtils; +use Psr\Http\Message\StreamFactoryInterface; class PubNub implements LoggerAwareInterface { - protected const SDK_VERSION = "7.4.0"; + protected const SDK_VERSION = "8.0.0"; protected const SDK_NAME = "PubNub-PHP"; public static $MAX_SEQUENCE = 65535; @@ -79,6 +85,15 @@ class PubNub implements LoggerAwareInterface protected LoggerInterface $logger; + /** + * PSR-7 compatible HTTP Client. Defaults to new GuzzleHttp\Client instance. + * + * @var ClientInterface + */ + protected ClientInterface $httpClient; + + protected RequestFactoryInterface $requestFactory; + protected int $nextSequence = 0; protected ?CryptoModule $cryptoModule = null; @@ -98,6 +113,9 @@ public function __construct(PNConfiguration $config) $this->telemetryManager = new TelemetryManager(); $this->tokenManager = new TokenManager(); $this->logger = new NullLogger(); + + $this->httpClient = new GuzzleHttpClient(); + $this->requestFactory = new GuzzleRequestFactory(); } /** @@ -510,6 +528,48 @@ public function setLogger(LoggerInterface $logger): void $this->logger = $logger; } + /** + * Set PSR-7 compatible HTTP Client + * + * @param ClientInterface $httpClient + * @return void + */ + public function setClient(ClientInterface $httpClient): void + { + $this->httpClient = $httpClient; + } + + /** + * Get PSR-7 compatible HTTP Client + * + * @return ClientInterface + */ + public function getClient(): ClientInterface + { + return $this->httpClient; + } + + /** + * Set PSR-7 compatible Request Factory + * + * @param RequestFactoryInterface $requestFactory + * @return void + */ + public function setRequestFactory(RequestFactoryInterface $requestFactory): void + { + $this->requestFactory = $requestFactory; + } + + /** + * Get PSR-7 compatible Request Factory + * + * @return RequestFactoryInterface + */ + public function getRequestFactory(): RequestFactoryInterface + { + return $this->requestFactory; + } + /** * @return GetState */ diff --git a/tests/Acceptance/PubNubContext.php b/tests/Acceptance/PubNubContext.php index b8c7b4e9..3d179d53 100644 --- a/tests/Acceptance/PubNubContext.php +++ b/tests/Acceptance/PubNubContext.php @@ -7,11 +7,16 @@ class PubNubContext { + protected int $retryLimit = 12; + /** @BeforeScenario */ public function before(BeforeScenarioScope $scope): void { + $this->waitForServer(); $ch = curl_init(); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); + curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 10); + foreach ($scope->getScenario()->getTags() as $tag) { if (0 === strpos($tag, 'contract')) { list(, $contractName) = explode('=', $tag); @@ -25,8 +30,11 @@ public function before(BeforeScenarioScope $scope): void /** @AfterScenario */ public function after(AfterScenarioScope $scope): void { + $this->waitForServer(); $ch = curl_init(); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); + curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 10); + foreach ($scope->getScenario()->getTags() as $tag) { if (0 === strpos($tag, 'contract')) { curl_setopt($ch, CURLOPT_URL, 'http://localhost:8090/expect'); @@ -35,4 +43,30 @@ public function after(AfterScenarioScope $scope): void } curl_close($ch); } + + protected function waitForServer(): void + { + + for ($i = 1; $i <= $this->retryLimit; $i++) { + print("Trying to connect ($i/$this->retryLimit)...\n"); + $ch = curl_init(); + curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); + curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 10); + curl_setopt($ch, CURLOPT_TIMEOUT, 10); + curl_setopt($ch, CURLOPT_URL, 'http://localhost:8090'); + curl_exec($ch); + $err = curl_error($ch); + if ($err === "") { + print("Server started\n"); + break; + } + print($err . "\n"); + + if ($i === $this->retryLimit) { + print("Server not started\n"); + exit(1); + } + sleep($i); + } + } } diff --git a/tests/PubNubTestCase.php b/tests/PubNubTestCase.php index 34bdf42e..064b3845 100755 --- a/tests/PubNubTestCase.php +++ b/tests/PubNubTestCase.php @@ -7,6 +7,7 @@ use Monolog\Logger; use Monolog\Handler\ErrorLogHandler; +// phpcs:ignore PSR1.Classes.ClassDeclaration abstract class PubNubTestCase extends TestCase { protected const CIPHER_KEY = "enigma"; @@ -29,16 +30,20 @@ abstract class PubNubTestCase extends TestCase /** @var PNConfiguration config */ protected $config; - /** @var PNConfiguration config */ + /** @var PNConfiguration config_demo */ + protected $config_demo; + + /** @var PNConfiguration config_enc */ protected $config_enc; - /** @var PNConfiguration config */ + /** @var PNConfiguration config_pam */ protected $config_pam; /** @var string */ protected $encodedSdkName; - protected function fakeSignature($params, $httpMethod, $timestamp, $publishKey, $path, $secretKey) { + protected function fakeSignature($params, $httpMethod, $timestamp, $publishKey, $path, $secretKey) + { $params['timestamp'] = (string) $timestamp; @@ -94,10 +99,16 @@ public function setUp(): void $this->config_pam->setSecretKey($secretKeyPam); $this->config_pam->setUuid($uuidMock); + $this->config_demo = new PNConfiguration(); + $this->config_demo->setSubscribeKey('demo'); + $this->config_demo->setPublishKey('demo'); + $this->config_demo->setUuid($uuidMock); + $this->config_demo->disableImmutableCheck(); + $this->pubnub = new PubNub($this->config); $this->pubnub_enc = new PubNub($this->config_enc); $this->pubnub_pam = new PubNub($this->config_pam); - $this->pubnub_demo = PubNub::demo(); + $this->pubnub_demo = new PubNub($this->config_demo); $this->pubnub->setLogger($logger); $this->pubnub_enc->setLogger($logger); diff --git a/tests/bootstrap.php b/tests/bootstrap.php index c4a002ff..fdf2fcb5 100755 --- a/tests/bootstrap.php +++ b/tests/bootstrap.php @@ -6,8 +6,6 @@ error_reporting(E_ALL); require_once(__DIR__ . '/PubNubTestCase.php'); -require_once(__DIR__ . '/helpers/Stub.php'); -require_once(__DIR__ . '/helpers/StubTransport.php'); if (!class_exists('Thread')) { class Thread diff --git a/tests/functional/MessageCountTest.php b/tests/functional/MessageCountTest.php index 039ce21f..44ded263 100644 --- a/tests/functional/MessageCountTest.php +++ b/tests/functional/MessageCountTest.php @@ -1,24 +1,21 @@ pubnub = new PubNub($this->config); - } - - public function testSyncDisabled() + public function testSyncDisabled(): void { $this->expectException(PubNubServerException::class); @@ -28,32 +25,32 @@ public function testSyncDisabled() "subscribe key.Login to your PubNub Dashboard Account and enable Storage & Playback.Contact support " . "@pubnub.com if you require further assistance.\",0,0]"; - $messageCount->stubFor("/v3/history/sub-key/". $this->pubnub->getConfiguration()->getSubscribeKey() . "/message-counts/my_channel") - ->withQuery([ + $messageCount->stubFor( + "/v3/history/sub-key/" . $this->pubnub->getConfiguration()->getSubscribeKey() . "/message-counts/my_channel" + )->withQuery([ "timetoken" => "10000", - "pnsdk" => $this->encodedSdkName, + "pnsdk" => $this->pubnub->getSdkFullName(), "uuid" => "myUUID", ]) ->setResponseBody($payload); $this->pubnub->getConfiguration()->setUuid("myUUID"); - $messageCount->channels(["my_channel"]) - ->channelsTimetoken(["10000"])->sync(); - + $messageCount->channels(["my_channel"])->channelsTimetoken(["10000"])->sync(); } - public function testSingleChannel_withSingleTimestamp() + public function testSingleChannelWithSingleTimestamp(): void { $messageCount = new MessageCountExposed($this->pubnub); $payload = "{\"status\": 200, \"error\": false, \"error_message\": \"\", " . "\"channels\": {\"my_channel\":19}}"; - $messageCount->stubFor("/v3/history/sub-key/". $this->pubnub->getConfiguration()->getSubscribeKey() . "/message-counts/my_channel") - ->withQuery([ + $messageCount->stubFor( + "/v3/history/sub-key/" . $this->pubnub->getConfiguration()->getSubscribeKey() . "/message-counts/my_channel" + )->withQuery([ "timetoken" => "10000", - "pnsdk" => $this->encodedSdkName, + "pnsdk" => $this->pubnub->getSdkFullName(), "uuid" => "myUUID", ]) ->setResponseBody($payload); @@ -66,14 +63,13 @@ public function testSingleChannel_withSingleTimestamp() $this->assertEquals(count($response->getChannels()), 1); $this->assertFalse(isset($response->getChannels()["channel_dont_exist"])); $this->assertTrue(isset($response->getChannels()["my_channel"])); - foreach($response->getChannels() as $channel => $count) { + foreach ($response->getChannels() as $channel => $count) { $this->assertEquals("my_channel", $channel); $this->assertEquals(19, $count); } - } - public function testSingleChannel_withMultiTimestamp() + public function testSingleChannelWithMultiTimestamp(): void { $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("The number of channels and the number of timetokens do not match"); @@ -82,20 +78,21 @@ public function testSingleChannel_withMultiTimestamp() $messageCount->channels(["my_channel"]) ->channelsTimetoken(["10000", "20000"])->sync(); - } - public function testMultiChannel_withSingleTimestamp() + public function testMultiChannelWithSingleTimestamp(): void { $messageCount = new MessageCountExposed($this->pubnub); $payload = "{\"status\": 200, \"error\": false, \"error_message\": \"\", " . "\"channels\": {\"my_channel\":19, \"new_channel\":5}}"; - $messageCount->stubFor("/v3/history/sub-key/". $this->pubnub->getConfiguration()->getSubscribeKey() . "/message-counts/my_channel,new_channel") - ->withQuery([ + $messageCount->stubFor( + "/v3/history/sub-key/" . $this->pubnub->getConfiguration()->getSubscribeKey() + . "/message-counts/my_channel,new_channel" + )->withQuery([ "timetoken" => "10000", - "pnsdk" => $this->encodedSdkName, + "pnsdk" => $this->pubnub->getSdkFullName(), "uuid" => "myUUID", ]) ->setResponseBody($payload); @@ -109,28 +106,28 @@ public function testMultiChannel_withSingleTimestamp() $this->assertFalse(isset($response->getChannels()["channel_dont_exist"])); $this->assertTrue(isset($response->getChannels()["my_channel"])); $this->assertTrue(isset($response->getChannels()["new_channel"])); - foreach($response->getChannels() as $channel => $count) { + foreach ($response->getChannels() as $channel => $count) { if ($channel === "my_channel") { $this->assertEquals(19, $count); - } - else if ($channel === "new_channel") { + } elseif ($channel === "new_channel") { $this->assertEquals(5, $count); } } - } - public function testMultiChannel_withMultiTimestamp() + public function testMultiChannelWithMultiTimestamp(): void { $messageCount = new MessageCountExposed($this->pubnub); $payload = "{\"status\": 200, \"error\": false, \"error_message\": \"\", " . - "\"channels\": {\"my_channel\":19, \"new_channel\":5}}"; + "\"channels\": {\"my_channel\":19, \"new_channel\":5}}"; - $messageCount->stubFor("/v3/history/sub-key/". $this->pubnub->getConfiguration()->getSubscribeKey() . "/message-counts/my_channel,new_channel") - ->withQuery([ + $messageCount->stubFor( + "/v3/history/sub-key/" . $this->pubnub->getConfiguration()->getSubscribeKey() + . "/message-counts/my_channel,new_channel" + )->withQuery([ "channelsTimetoken" => PubNubUtil::joinitems(["10000", "20000"]), - "pnsdk" => $this->encodedSdkName, + "pnsdk" => $this->pubnub->getSdkFullName(), "uuid" => "myUUID", ]) ->setResponseBody($payload); @@ -144,18 +141,16 @@ public function testMultiChannel_withMultiTimestamp() $this->assertFalse(isset($response->getChannels()["channel_dont_exist"])); $this->assertTrue(isset($response->getChannels()["my_channel"])); $this->assertTrue(isset($response->getChannels()["new_channel"])); - foreach($response->getChannels() as $channel => $count) { + foreach ($response->getChannels() as $channel => $count) { if ($channel === "my_channel") { $this->assertEquals(19, $count); - } - else if ($channel === "new_channel") { + } elseif ($channel === "new_channel") { $this->assertEquals(5, $count); } } - } - public function testWithoutTimeToken() + public function testWithoutTimeToken(): void { $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Timetoken missing"); @@ -163,10 +158,9 @@ public function testWithoutTimeToken() $messageCount = new MessageCountExposed($this->pubnub); $messageCount->channels(["my_channel"])->sync(); - } - public function testWithoutChannels_SingleTimeToken() + public function testWithoutChannelsSingleTimeToken(): void { $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channel missing"); @@ -174,10 +168,9 @@ public function testWithoutChannels_SingleTimeToken() $messageCount = new MessageCountExposed($this->pubnub); $messageCount->channelsTimetoken(["10000"])->sync(); - } - public function testWithoutChannels_TimeTokenList() + public function testWithoutChannelsTimeTokenList(): void { $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channel missing"); @@ -185,20 +178,20 @@ public function testWithoutChannels_TimeTokenList() $messageCount = new MessageCountExposed($this->pubnub); $messageCount->channelsTimetoken(["10000", "20000"])->sync(); - } - public function testChannel_withMultiEmptyToken() + public function testChannelWithMultiEmptyToken(): void { $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Timetoken missing"); $messageCount = new MessageCountExposed($this->pubnub); - $messageCount->stubFor("/v3/history/sub-key/". $this->pubnub->getConfiguration()->getSubscribeKey() . "/message-counts/my_channel") - ->withQuery([ + $messageCount->stubFor( + "/v3/history/sub-key/" . $this->pubnub->getConfiguration()->getSubscribeKey() . "/message-counts/my_channel" + )->withQuery([ "channelsToken" => PubNubUtil::joinitems([]), - "pnsdk" => $this->encodedSdkName, + "pnsdk" => $this->pubnub->getSdkFullName(), "uuid" => "myUUID", ]); @@ -206,20 +199,20 @@ public function testChannel_withMultiEmptyToken() $messageCount->channels(["my_channel"]) ->channelsTimetoken([])->sync(); - } - public function testChannel_withMultiNullToken() + public function testChannelWithMultiNullToken(): void { $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Timetoken missing"); $messageCount = new MessageCountExposed($this->pubnub); - $messageCount->stubFor("/v3/history/sub-key/". $this->pubnub->getConfiguration()->getSubscribeKey() . "/message-counts/my_channel") - ->withQuery([ + $messageCount->stubFor( + "/v3/history/sub-key/" . $this->pubnub->getConfiguration()->getSubscribeKey() . "/message-counts/my_channel" + )->withQuery([ "timetoken" => null, - "pnsdk" => $this->encodedSdkName, + "pnsdk" => $this->pubnub->getSdkFullName(), "uuid" => "myUUID", ]); @@ -227,32 +220,25 @@ public function testChannel_withMultiNullToken() $messageCount->channels(["my_channel"]) ->channelsTimetoken(null)->sync(); - } - } +// phpcs:ignore PSR1.Classes.ClassDeclaration class MessageCountExposed extends MessageCount { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function requestOptions() + public function stubFor(string $url): PsrStub { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } - -} \ No newline at end of file +} diff --git a/tests/functional/push/ListPushProvisionsTest.php b/tests/functional/push/ListPushProvisionsTest.php index e0ae5733..f3d0de1f 100644 --- a/tests/functional/push/ListPushProvisionsTest.php +++ b/tests/functional/push/ListPushProvisionsTest.php @@ -7,7 +7,6 @@ use PubNub\PubNub; use PubNub\PubNubUtil; use PubNubTestCase; -use Tests\Helpers\StubTransport; class ListPushProvisionsTest extends PubNubTestCase { @@ -106,20 +105,6 @@ public function testListChannelGroupMPNS() // phpcs:ignore PSR1.Classes.ClassDeclaration class ListPushProvisionsExposed extends ListPushProvisions { - protected $transport; - - public function __construct(PubNub $pubnubInstance) - { - parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); - } - public function buildParams() { return parent::buildParams(); @@ -129,11 +114,4 @@ public function buildPath() { return parent::buildPath(); } - - public function requestOptions() - { - return [ - 'transport' => $this->transport - ]; - } } diff --git a/tests/helpers/PsrStub.php b/tests/helpers/PsrStub.php new file mode 100644 index 00000000..7a3d6604 --- /dev/null +++ b/tests/helpers/PsrStub.php @@ -0,0 +1,77 @@ +path = $path; + } + + /** + * @param String[] $query + * @return PsrStub + */ + public function withQuery(array $query): self + { + $this->query = http_build_query($query); + $this->query = str_replace(['%2C', '%25'], [',', '%'], $this->query); + return $this; + } + + public function setResponseBody(string $responseBody): self + { + $this->responseBody = $responseBody; + return $this; + } + + public function setResponseStatus(int $responseStatus): self + { + $this->responseStatus = $responseStatus; + return $this; + } + + public function setResponseHeaders(mixed $responseHeaders): self + { + $this->responseHeaders = $responseHeaders; + return $this; + } + + public function isPathMatch(string $path): bool + { + return $this->path === $path; + } + + public function isQueryMatch(string $query): bool + { + $expected = []; + $actual = []; + parse_str($this->query, $expected); + ksort($expected); + parse_str($query, $actual); + ksort($actual); + $expected = http_build_query($expected); + $actual = http_build_query($actual); + + return $expected === $actual; + } + + public function getResponse(): ResponseInterface + { + if (isset($this->responseStatus)) { + return new Response($this->responseStatus, ['Content-Type' => 'application/json'], $this->responseBody); + } + $headers = $this->responseHeaders ?? ['Content-Type' => 'application/json']; + return new Response(200, $headers, $this->responseBody); + } +} diff --git a/tests/helpers/PsrStubClient.php b/tests/helpers/PsrStubClient.php new file mode 100644 index 00000000..e3e68f94 --- /dev/null +++ b/tests/helpers/PsrStubClient.php @@ -0,0 +1,47 @@ +addStub($stub); + return $stub; + } + + public function addStub(PsrStub $stub): self + { + $this->stubs[] = $stub; + return $this; + } + + public function sendRequest(RequestInterface $request): ResponseInterface + { + $pathFound = false; + $queryFound = false; + + foreach ($this->stubs as $stub) { + $pathFound = $pathFound || $stub->isPathMatch($request->getUri()->getPath()); + $queryFound = $queryFound || $stub->isQueryMatch($request->getUri()->getQuery()); + + if ($pathFound && $queryFound) { + return $stub->getResponse(); + } + } + throw new \Exception( + "No stub matched for:" . $request->getUri()->getPath() . "?" . $request->getUri()->getQuery() . + (!$pathFound ? "PathNotFound" : "") . (!$queryFound ? "QueryNotFound" : "") + ); + } +} diff --git a/tests/helpers/Stub.php b/tests/helpers/Stub.php deleted file mode 100644 index 79cd08ec..00000000 --- a/tests/helpers/Stub.php +++ /dev/null @@ -1,182 +0,0 @@ -path = $path; - } - - public function __toString() - { - $queryString = $this->queryString(); - - if ($queryString) { - return $this->path . '?' . $this->queryString(); - } else { - return $this->path; - } - } - - /** - * @return string - */ - public function getBody() - { - return $this->body; - } - - /** - * @return string - */ - public function getStatus() - { - return $this->status; - } - - /** - * @param string $body - * @return $this - */ - public function setResponseBody($body) - { - $this->body = $body; - - return $this; - } - - /** - * @param string $status - * @return $this - * @throws \Exception - */ - public function setResponseStatus($status) - { - if (!is_string($status)) { - throw new \Exception("Stubbed status should be a string like \"HTTP/1.1 200 OK\""); - } - - $this->status = $status; - - return $this; - } - - /** - * @return $this - */ - public function usePost() - { - $this->method = "POST"; - - return $this; - } - - /** - * @param string | array $query - * @return $this - */ - public function withQuery($query) - { - if (is_string($query)) { - $queryArray = []; - - parse_str($query, $queryArray); - - $this->query = $queryArray; - } else { - $this->query = $query; - } - - return $this; - } - - private function stripKeys($path) - { - $patterns = ['#\/(demo|sub-c-[a-z0-9-]{36})\/#']; - $replaces = ["/{SUB_KEY}/"]; - return preg_replace($patterns, $replaces, $path); - } - - public function isPathMatch($path) - { - return $this->stripKeys($this->path) === $this->stripKeys($path); - } - - public function isQueryMatch($actualQueryString) - { - $actualQuery = []; - $queryElements = explode("&", $actualQueryString); - - foreach ($queryElements as $element) { - $keyVal = explode("=", $element); - $actualQuery[$keyVal[0]] = $keyVal[1]; - } - - foreach ($actualQuery as $key => $value) { - if (array_key_exists($key, $this->query)) { - if ($this->query[$key] === static::ANY) { - continue; - } - - if ($this->query[$key] !== $value) { - return false; - } - } else { - return false; - } - } - - foreach ($this->query as $key => $value) { - if (!array_key_exists($key, $actualQuery)) { - return false; - } - } - - return true; - } - - public function queryString() - { - $queryArray = []; - - foreach ($this->query as $key => $value) { - if ($value == static::ANY) { - $value = "{ANY}"; - }; - - $queryArray[] = "$key=$value"; - } - - return join("&", $queryArray); - } -} - -// phpcs:ignore PSR1.Classes.ClassDeclaration -class StubException extends \Exception -{ -} diff --git a/tests/helpers/StubTransport.php b/tests/helpers/StubTransport.php deleted file mode 100644 index 3de975f0..00000000 --- a/tests/helpers/StubTransport.php +++ /dev/null @@ -1,76 +0,0 @@ -requestsCountInt++; - - $parsedUrl = parse_url($url); - - foreach ($this->stubs as $stub) { - if ($stub->isPathMatch($parsedUrl["path"]) && $stub->isQueryMatch($parsedUrl['query'])) { - $response = $stub->getStatus(); - $response .= "Content-Type: text/plain\r\n"; - $response .= "Connection: close\r\n\r\n"; - $response .= $stub->getBody(); - - return $response; - } - } - - $response = "HTTP/1.0 530 No Stub Matched\r\n"; - $response .= "Content-Type: text/plain\r\n"; - $response .= "Connection: close\r\n\r\n"; - - $response .= "No stubs matched for request:\n"; - $response .= $parsedUrl['path'] . "?" . $parsedUrl['query'] . "\n\n"; - $response .= "Existing stubs:\n" . join("\n", $this->stubs) . "\n\n"; - - return $response; - } - - /** - * @return int - */ - public function requestsCount() - { - return $this->requestsCountInt; - } - - /** - * @param string $path - * @return Stub - */ - public function stubFor($path) - { - $stub = new Stub($path); - - $this->stubs[] = $stub; - - return $stub; - } - - // phpcs:ignore PSR1.Methods.CamelCapsMethodName - public function request_multiple($requests, $options) - { - throw new \Exception("Not implemented"); - } - - public static function test($capabilities = []) - { - return true; - } -} diff --git a/tests/integrational/AddChannelChannelGroupEndpointTest.php b/tests/integrational/AddChannelChannelGroupEndpointTest.php index b100f4e2..c57d9560 100644 --- a/tests/integrational/AddChannelChannelGroupEndpointTest.php +++ b/tests/integrational/AddChannelChannelGroupEndpointTest.php @@ -7,23 +7,24 @@ use PubNub\Exceptions\PubNubValidationException; use PubNub\PubNub; use PubNubTestCase; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class AddChannelChannelGroupEndpointTest extends PubNubTestCase { public function testSuccess() { - $addChannelChannelGroup = new AddChannelChannelGroupExposed($this->pubnub); + $addChannelChannelGroup = new AddChannelChannelGroupExposed($this->pubnub_demo); $addChannelChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/gr%7CoupA") ->withQuery([ - "pnsdk" => $this->encodedSdkName, + "pnsdk" => $this->pubnub_demo->getSdkFullName(), "uuid" => "myUUID", - "add" => "c%7Ch1,ch2s" + "add" => "c%7Ch1,ch2s", ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {},\"service\":\"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); $response = $addChannelChannelGroup->channelGroup("gr|oupA")->channels(["c|h1", "ch2s"])->sync(); @@ -35,17 +36,17 @@ public function testGroupMissing() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channel group missing"); - $addChannelChannelGroup = new AddChannelChannelGroupExposed($this->pubnub); + $addChannelChannelGroup = new AddChannelChannelGroupExposed($this->pubnub_demo); $addChannelChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/gr%7CoupA") ->withQuery([ - "pnsdk" => $this->encodedSdkName, + "pnsdk" => $this->pubnub_demo->getSdkFullName(), "uuid" => "myUUID", "add" => "c%7Ch1,ch2s" ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {},\"service\":\"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); $addChannelChannelGroup->channels(["c|h1", "ch2s"])->sync(); } @@ -55,17 +56,17 @@ public function testGroupIsEmpty() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channel group missing"); - $addChannelChannelGroup = new AddChannelChannelGroupExposed($this->pubnub); + $addChannelChannelGroup = new AddChannelChannelGroupExposed($this->pubnub_demo); $addChannelChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/gr%7CoupA") ->withQuery([ - "pnsdk" => $this->encodedSdkName, + "pnsdk" => $this->pubnub_demo->getSdkFullName(), "uuid" => "myUUID", "add" => "c%7Ch1,ch2s" ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {},\"service\":\"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); $addChannelChannelGroup->channelGroup("")->channels(["c|h1", "ch2s"])->sync(); } @@ -74,17 +75,16 @@ public function testChannelMissing() { $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channels missing"); - - $addChannelChannelGroup = new AddChannelChannelGroupExposed($this->pubnub); + $addChannelChannelGroup = new AddChannelChannelGroupExposed($this->pubnub_demo); $addChannelChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/groupA") ->withQuery([ - "pnsdk" => $this->encodedSdkName, + "pnsdk" => $this->pubnub_demo->getSdkFullName(), "uuid" => "myUUID" ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {},\"service\":\"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); $addChannelChannelGroup->channelGroup("groupA")->sync(); } @@ -92,18 +92,18 @@ public function testChannelMissing() public function testIsAuthRequiredSuccess() { $this->expectNotToPerformAssertions(); - $addChannelChannelGroup = new AddChannelChannelGroupExposed($this->pubnub); + $addChannelChannelGroup = new AddChannelChannelGroupExposed($this->pubnub_demo); $addChannelChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/groupA") ->withQuery([ - "pnsdk" => $this->encodedSdkName, + "pnsdk" => $this->pubnub_demo->getSdkFullName(), "uuid" => "myUUID", "auth" => "myKey", "add" => "ch1,ch2" ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {},\"service\":\"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID")->setAuthKey("myKey"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID")->setAuthKey("myKey"); $addChannelChannelGroup->channelGroup("groupA")->channels(["ch1", "ch2"])->sync(); } @@ -113,19 +113,19 @@ public function testErrorBodyForbidden() $this->expectException(PubNubServerException::class); $this->expectExceptionMessage("Server responded with an error and the status code is 403"); - $addChannelChannelGroup = new AddChannelChannelGroupExposed($this->pubnub); + $addChannelChannelGroup = new AddChannelChannelGroupExposed($this->pubnub_demo); $addChannelChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/groupA") ->withQuery([ - "pnsdk" => $this->encodedSdkName, + "pnsdk" => $this->pubnub_demo->getSdkFullName(), "uuid" => "myUUID", "auth" => "myKey", "add" => "ch1,ch2" ]) - ->setResponseStatus("HTTP/1.0 403 Forbidden") + ->setResponseStatus(403) ->setResponseBody("{\"status\": 403, \"message\": \"OK\", \"payload\": {},\"service\":\"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID")->setAuthKey("myKey"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID")->setAuthKey("myKey"); $addChannelChannelGroup->channelGroup("groupA")->channels(["ch1", "ch2"])->sync(); } @@ -151,24 +151,19 @@ public function testSuperCallTest() // phpcs:ignore PSR1.Classes.ClassDeclaration class AddChannelChannelGroupExposed extends AddChannelToChannelGroup { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function requestOptions() + public function stubFor(string $url): PsrStub { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } diff --git a/tests/integrational/FetchMessagesTest.php b/tests/integrational/FetchMessagesTest.php index ddf4cc93..0eab7ea0 100644 --- a/tests/integrational/FetchMessagesTest.php +++ b/tests/integrational/FetchMessagesTest.php @@ -6,7 +6,8 @@ use PubNub\Models\Consumer\MessagePersistence\PNFetchMessagesResult; use PubNub\PubNub; use PubNubTestCase; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class FetchMessagesTest extends PubNubTestCase { @@ -28,12 +29,12 @@ public function testFetchWithDefaults() $fetchMessages ->stubFor("/v3/history/sub-key/demo/channel/TheMessageHistoryChannelHD") ->withQuery([ - "uuid" => $this->pubnub_demo->getConfiguration()->getUserId(), - "pnsdk" => $this->encodedSdkName, "include_meta" => "false", "include_uuid" => "false", "include_message_type" => "true", "include_custom_message_type" => "false", + "pnsdk" => $this->encodedSdkName, + "uuid" => $this->pubnub_demo->getConfiguration()->getUserId(), ]) ->setResponseBody('{"status": 200, "error": false, "error_message": "", "channels": {"TheMessageHistoryChannelHD":[ @@ -67,12 +68,12 @@ public function testFetchWithCount() ->stubFor("/v3/history/sub-key/{$subKey}/channel/TheMessageHistoryChannelHD") ->withQuery([ "max" => "5", - "uuid" => $this->pubnub_demo->getConfiguration()->getUserId(), - "pnsdk" => $this->encodedSdkName, "include_meta" => "false", "include_uuid" => "false", "include_message_type" => "true", "include_custom_message_type" => "false", + "pnsdk" => $this->encodedSdkName, + "uuid" => $this->pubnub_demo->getConfiguration()->getUserId(), ]) ->setResponseBody('{"status": 200, "error": false, "error_message": "", "channels": {"TheMessageHistoryChannelHD":[ @@ -102,12 +103,12 @@ public function testFetchWithStartEnd() ->withQuery([ "start" => "17165627042258346", "end" => "17165627042258546", - "uuid" => $this->pubnub_demo->getConfiguration()->getUserId(), - "pnsdk" => $this->encodedSdkName, "include_meta" => "false", "include_uuid" => "false", "include_message_type" => "true", "include_custom_message_type" => "false", + "pnsdk" => $this->encodedSdkName, + "uuid" => $this->pubnub_demo->getConfiguration()->getUserId(), ]) ->setResponseBody('{"status": 200, "error": false, "error_message": "", "channels": {"TheMessageHistoryChannelHD":[ @@ -135,12 +136,12 @@ public function testFetchEncrypted() $fetchMessages ->stubFor("/v3/history/sub-key/{$subKey}/channel/TheMessageHistoryChannelHD-ENCRYPTED") ->withQuery([ - "uuid" => $this->pubnub_enc->getConfiguration()->getUserId(), - "pnsdk" => $this->encodedSdkName, "include_meta" => "false", "include_uuid" => "false", "include_message_type" => "true", "include_custom_message_type" => "false", + "pnsdk" => $this->encodedSdkName, + "uuid" => $this->pubnub_enc->getConfiguration()->getUserId(), ]) ->setResponseBody('{"status": 200, "error": false, "error_message": "", "channels": { @@ -166,24 +167,19 @@ public function testFetchEncrypted() // phpcs:ignore PSR1.Classes.ClassDeclaration class FetchMessagesExposed extends FetchMessages { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function requestOptions() + public function stubFor(string $url): PsrStub { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } diff --git a/tests/integrational/FilesTest.php b/tests/integrational/FilesTest.php index 90366d4d..38321591 100644 --- a/tests/integrational/FilesTest.php +++ b/tests/integrational/FilesTest.php @@ -2,7 +2,12 @@ namespace PubNubTests\integrational; +use PubNub\PNConfiguration; +use PubNub\PubNub; use PubNubTestCase; +use PubNubTests\helpers\PsrStubClient; +use PubNub\Exceptions\PubNubResponseParsingException; +use PubNub\Exceptions\PubNubServerException; class FilesTest extends PubNubTestCase { @@ -102,4 +107,36 @@ public function testDeleteAllFiles() $this->assertEquals(200, $response->getStatus(), "Unexpected status value"); } } + + public function testEmptyFileListAfterDelete(): void + { + $response = $this->pubnub->listFiles()->channel($this->channel)->sync(); + $this->assertNotEmpty($response); + $this->assertCount(0, $response->getData()); + } + + public function testThrowErrorOnMalformedResponse(): void + { + $this->expectException(PubNubResponseParsingException::class); + $client = new PsrStubClient(); + $config = new PNConfiguration(); + $config->setPublishKey("demo"); + $config->setSubscribeKey("demo"); + $config->setUserId("test"); + $pubnub = new PubNub($config); + $pubnub->setClient($client); + $client->stubFor("/v1/files/demo/channels/files-test/files/none/none") + ->withQuery(["pnsdk" => $pubnub->getSdkFullName(), "uuid" => "test"]) + ->setResponseBody('{}') + ->setResponseStatus(307) + ->setResponseHeaders(['Location' => '']); + $this->pubnub->setClient($client); + $pubnub->getFileDownloadUrl()->channel($this->channel)->fileId('none')->fileName('none')->sync(); + } + + public function testThrowErrorOnNoFileFound(): void + { + $this->expectException(PubNubServerException::class); + $this->pubnub->downloadFile()->channel($this->channel)->fileId('-')->fileName('-')->sync(); + } } diff --git a/tests/integrational/GetStateTest.php b/tests/integrational/GetStateTest.php index e0fb3cb1..2061672c 100644 --- a/tests/integrational/GetStateTest.php +++ b/tests/integrational/GetStateTest.php @@ -8,23 +8,24 @@ use PubNub\PubNub; use PubNub\PubNubUtil; use PubNubTestCase; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class GetStateTest extends PubNubTestCase { public function testOneChannel() { - $getState = new GetStateExposed($this->pubnub); + $getState = new GetStateExposed($this->pubnub_demo); $getState->stubFor("/v2/presence/sub-key/demo/channel/testChannel/uuid/sampleUUID") ->withQuery([ + "pnsdk" => $this->encodedSdkName, "uuid" => "sampleUUID", - "pnsdk" => $this->encodedSdkName ]) ->setResponseBody("{ \"status\": 200, \"message\": \"OK\", " . " \"payload\": { \"age\" : 20, \"status\" : \"online\"}, \"service\": \"Presence\"}"); - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("sampleUUID"); $response = $getState->channels("testChannel")->sync(); @@ -34,7 +35,7 @@ public function testOneChannel() public function testOneChannelWithoutUUID() { - $getState = new GetStateExposed($this->pubnub); + $getState = new GetStateExposed($this->pubnub_demo); $getState->stubFor("/v2/presence/sub-key/demo/channel/testChannel/uuid/myUUID") ->withQuery([ @@ -44,7 +45,7 @@ public function testOneChannelWithoutUUID() ->setResponseBody("{ \"status\": 200, \"message\": \"OK\", " . "\"payload\": { \"age\" : 20, \"status\" : \"online\"}, \"service\": \"Presence\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); $response = $getState->channels("testChannel")->sync(); @@ -56,34 +57,35 @@ public function testFailedPayload() { $this->expectException(PubNubException::class); - $getState = new GetStateExposed($this->pubnub); + $getState = new GetStateExposed($this->pubnub_demo); $getState->stubFor("/v2/presence/sub-key/demo/channel/testChannel/uuid/sampleUUID") ->withQuery([ + "pnsdk" => $this->encodedSdkName, "uuid" => "sampleUUID", ]) ->setResponseBody("{ \"status\": 200, \"message\": \"OK\", " . "\"payload\": { \"age\" : 20, \"status\" : \"online\"}, \"service\": \"Presence\"}"); - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("sampleUUID"); - $getState->channels("testChannel")->sync(); + $getState->sync(); } public function testMultipleChannel() { - $getState = new GetStateExposed($this->pubnub); + $getState = new GetStateExposed($this->pubnub_demo); $getState->stubFor("/v2/presence/sub-key/demo/channel/ch1,ch2/uuid/sampleUUID") ->withQuery([ + "pnsdk" => $this->encodedSdkName, "uuid" => "sampleUUID", - "pnsdk" => $this->encodedSdkName ]) ->setResponseBody("{ \"status\": 200, \"message\": \"OK\", " . "\"payload\": { \"channels\": { \"ch1\": { \"age\" : 20, \"status\" : \"online\"}, " . "\"ch2\": { \"age\": 100, \"status\": \"offline\" } } }, \"service\": \"Presence\"}"); - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("sampleUUID"); $response = $getState->channels(["ch1", "ch2"])->sync(); @@ -96,19 +98,19 @@ public function testMultipleChannel() public function testOneChannelGroup() { - $getState = new GetStateExposed($this->pubnub); + $getState = new GetStateExposed($this->pubnub_demo); $getState->stubFor("/v2/presence/sub-key/demo/channel/,/uuid/sampleUUID") ->withQuery([ - "uuid" => "sampleUUID", + "channel-group" => "cg1", "pnsdk" => $this->encodedSdkName, - "channel-group" => "cg1" + "uuid" => "sampleUUID", ]) ->setResponseBody("{ \"status\": 200, \"message\": \"OK\", " . "\"payload\": { \"channels\": { \"chcg1\": { \"age\" : 20, \"status\" : \"online\"}, " . "\"chcg2\": { \"age\": 100, \"status\": \"offline\" } } }, \"service\": \"Presence\"}"); - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("sampleUUID"); $response = $getState->channelGroups("cg1")->sync(); @@ -121,19 +123,19 @@ public function testOneChannelGroup() public function testManyChannelGroup() { - $getState = new GetStateExposed($this->pubnub); + $getState = new GetStateExposed($this->pubnub_demo); $getState->stubFor("/v2/presence/sub-key/demo/channel/,/uuid/sampleUUID") ->withQuery([ - "uuid" => "sampleUUID", + "channel-group" => PubNubUtil::urlEncode("cg1,cg2"), "pnsdk" => $this->encodedSdkName, - "channel-group" => PubNubUtil::urlEncode("cg1,cg2") + "uuid" => "sampleUUID", ]) ->setResponseBody("{ \"status\": 200, \"message\": \"OK\", " . "\"payload\": { \"channels\": { \"chcg1\": { \"age\" : 20, \"status\" : \"online\"}, " . "\"chcg2\": { \"age\": 100, \"status\": \"offline\" } } }, \"service\": \"Presence\"}"); - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("sampleUUID"); $response = $getState->channelGroups(["cg1", "cg2"])->sync(); @@ -146,19 +148,19 @@ public function testManyChannelGroup() public function testCombination() { - $getState = new GetStateExposed($this->pubnub); + $getState = new GetStateExposed($this->pubnub_demo); $getState->stubFor("/v2/presence/sub-key/demo/channel/ch1/uuid/sampleUUID") ->withQuery([ - "uuid" => "sampleUUID", + "channel-group" => PubNubUtil::urlEncode("cg1,cg2"), "pnsdk" => $this->encodedSdkName, - "channel-group" => PubNubUtil::urlEncode("cg1,cg2") + "uuid" => "sampleUUID", ]) ->setResponseBody("{ \"status\": 200, \"message\": \"OK\", " . "\"payload\": { \"channels\": { \"chcg1\": { \"age\" : 20, \"status\" : \"online\"}, " . "\"chcg2\": { \"age\": 100, \"status\": \"offline\" } } }, \"service\": \"Presence\"}"); - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("sampleUUID"); $response = $getState->channels("ch1")->channelGroups(["cg1", "cg2"])->sync(); @@ -174,17 +176,17 @@ public function testMissingChannelAndGroup() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channel or group missing"); - $getState = new GetStateExposed($this->pubnub); + $getState = new GetStateExposed($this->pubnub_demo); $getState->stubFor("/v2/presence/sub-key/demo/channel/testChannel/uuid/sampleUUID") ->withQuery([ + "pnsdk" => $this->encodedSdkName, "uuid" => "sampleUUID", - "pnsdk" => $this->encodedSdkName ]) ->setResponseBody("{ \"status\": 200, \"message\": \"OK\", " . "\"payload\": { \"age\" : 20, \"status\" : \"online\"}, \"service\": \"Presence\"}"); - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("sampleUUID"); $getState->sync(); } @@ -198,12 +200,12 @@ public function testIsAuthRequiredSuccess() $pubnub = new PubNub($config); $getState = new GetStateExposed($pubnub); - - $getState->stubFor("/v2/presence/sub-key/demo/channel/testChannel/uuid/sampleUUID") + $subKey = $pubnub->getConfiguration()->getSubscribeKey(); + $getState->stubFor("/v2/presence/sub-key/{$subKey}/channel/testChannel/uuid/sampleUUID") ->withQuery([ - "uuid" => "sampleUUID", "pnsdk" => $this->encodedSdkName, - "auth" => "myKey" + "uuid" => "sampleUUID", + "auth" => "myKey", ]) ->setResponseBody("{ \"status\": 200, \"message\": \"OK\", " . "\"payload\": { \"age\" : 20, \"status\" : \"online\"}, \"service\": \"Presence\"}"); @@ -223,8 +225,8 @@ public function testNullSubKey() $getState->stubFor("/v2/presence/sub-key/demo/channel/testChannel/uuid/sampleUUID") ->withQuery([ + "pnsdk" => $this->encodedSdkName, "uuid" => "sampleUUID", - "pnsdk" => $this->encodedSdkName ]) ->setResponseBody("{ \"status\": 200, \"message\": \"OK\", " . "\"payload\": { \"age\" : 20, \"status\" : \"online\"}, \"service\": \"Presence\"}"); @@ -234,8 +236,6 @@ public function testNullSubKey() public function testEmptySubKeySync() { - - $this->expectException(PubNubException::class); $this->expectExceptionMessage("Subscribe Key not configured"); @@ -247,8 +247,8 @@ public function testEmptySubKeySync() $getState->stubFor("/v2/presence/sub-key/demo/channel/testChannel/uuid/sampleUUID") ->withQuery([ + "pnsdk" => $this->encodedSdkName, "uuid" => "sampleUUID", - "pnsdk" => $this->encodedSdkName ]) ->setResponseBody("{ \"status\": 200, \"message\": \"OK\", " . "\"payload\": { \"age\" : 20, \"status\" : \"online\"}, \"service\": \"Presence\"}"); @@ -272,24 +272,19 @@ public function testSuperCall() //phpcs:ignore PSR1.Classes.ClassDeclaration class GetStateExposed extends GetState { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function requestOptions() + public function stubFor(string $url): PsrStub { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } diff --git a/tests/integrational/HereNowTest.php b/tests/integrational/HereNowTest.php index 30a96793..79bdf48f 100644 --- a/tests/integrational/HereNowTest.php +++ b/tests/integrational/HereNowTest.php @@ -2,11 +2,11 @@ namespace Tests\Integrational; -use Countable; +use PubNub\PubNub; use PubNub\Endpoints\Presence\HereNow; use PubNub\Exceptions\PubNubValidationException; -use Tests\Helpers\Stub; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class HereNowTest extends \PubNubTestCase { @@ -16,12 +16,12 @@ class HereNowTest extends \PubNubTestCase */ public function testMultipleChannelState() { - $hereNow = new HereNowExposed($this->pubnub); + $hereNow = new HereNowExposed($this->pubnub_demo); $hereNow->stubFor("/v2/presence/sub-key/demo/channel/ch1,ch2") ->withQuery([ 'state' => '1', 'pnsdk' => $this->encodedSdkName, - 'uuid' => Stub::ANY + 'uuid' => $this->pubnub_demo->getConfiguration()->getUuid(), ]) ->setResponseBody("{\"status\":200,\"message\":\"OK\",\"payload\":{\"total_occupancy\":3,\"total_channels\"" . ":2,\"channels\":{\"ch1\":{\"occupancy\":1,\"uuids\":[{\"uuid\":\"user1\"}]},\"ch2\":{\"occupancy\":2" @@ -53,12 +53,12 @@ public function testMultipleChannelState() */ public function testMultipleChannel() { - $hereNow = new HereNowExposed($this->pubnub); + $hereNow = new HereNowExposed($this->pubnub_demo); $hereNow->stubFor("/v2/presence/sub-key/demo/channel/ch1,ch2") ->withQuery([ 'state' => '1', 'pnsdk' => $this->encodedSdkName, - 'uuid' => Stub::ANY + 'uuid' => $this->pubnub_demo->getConfiguration()->getUuid(), ]) ->setResponseBody("{\"status\":200,\"message\":\"OK\",\"payload\":{\"total_occupancy\":3,\"total_channels\"" . ":2,\"channels\":{\"ch1\":{\"occupancy\":1,\"uuids\":[{\"uuid\":\"user1\"}]},\"ch2\":{\"occupancy\":2" @@ -90,11 +90,11 @@ public function testMultipleChannel() */ public function testMultipleChannelWithoutState() { - $hereNow = new HereNowExposed($this->pubnub); + $hereNow = new HereNowExposed($this->pubnub_demo); $hereNow->stubFor("/v2/presence/sub-key/demo/channel/game1,game2") ->withQuery([ 'pnsdk' => $this->encodedSdkName, - 'uuid' => Stub::ANY + 'uuid' => $this->pubnub_demo->getConfiguration()->getUuid(), ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {\"channels\": {\"game1\": {\"uuids" . "\": [\"a3ffd012-a3b9-478c-8705-64089f24d71e\"], \"occupancy\": 1}}, \"total_channels\": 1, \"total_o" @@ -121,12 +121,12 @@ public function testMultipleChannelWithoutState() */ public function testMultipleChannelWithoutStateUUIDs() { - $hereNow = new HereNowExposed($this->pubnub); + $hereNow = new HereNowExposed($this->pubnub_demo); $hereNow->stubFor("/v2/presence/sub-key/demo/channel/game1,game2") ->withQuery([ 'disable-uuids' => '1', - 'uuid' => Stub::ANY, - 'pnsdk' => $this->encodedSdkName + 'pnsdk' => $this->encodedSdkName, + 'uuid' => $this->pubnub_demo->getConfiguration()->getUuid(), ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {\"channels\": {\"game1\": {\"occupa" . "ncy\": 1}}, \"total_channels\": 1, \"total_occupancy\": 1}, \"service\": \"Presence\"}"); @@ -147,12 +147,12 @@ public function testMultipleChannelWithoutStateUUIDs() */ public function testSingularChannelWithoutStateUUIDs() { - $hereNow = new HereNowExposed($this->pubnub); + $hereNow = new HereNowExposed($this->pubnub_demo); $hereNow->stubFor("/v2/presence/sub-key/demo/channel/game1") ->withQuery([ 'disable-uuids' => '1', - 'uuid' => Stub::ANY, - 'pnsdk' => $this->encodedSdkName + 'pnsdk' => $this->encodedSdkName, + 'uuid' => $this->pubnub_demo->getConfiguration()->getUuid(), ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"service\": \"Presence\", \"occupancy\": 3}"); @@ -168,11 +168,11 @@ public function testSingularChannelWithoutStateUUIDs() */ public function testSingularChannelWithoutState() { - $hereNow = new HereNowExposed($this->pubnub); + $hereNow = new HereNowExposed($this->pubnub_demo); $hereNow->stubFor("/v2/presence/sub-key/demo/channel/game1") ->withQuery([ - 'uuid' => Stub::ANY, - 'pnsdk' => $this->encodedSdkName + 'pnsdk' => $this->encodedSdkName, + 'uuid' => $this->pubnub_demo->getConfiguration()->getUuid(), ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"service\": \"Presence\", \"uuids\": [\"a3ffd012" . "-a3b9-478c-8705-64089f24d71e\"], \"occupancy\": 1}"); @@ -197,12 +197,12 @@ public function testSingularChannelWithoutState() */ public function testSingularChannel() { - $hereNow = new HereNowExposed($this->pubnub); + $hereNow = new HereNowExposed($this->pubnub_demo); $hereNow->stubFor("/v2/presence/sub-key/demo/channel/game1") ->withQuery([ 'state' => '1', - 'uuid' => Stub::ANY, - 'pnsdk' => $this->encodedSdkName + 'pnsdk' => $this->encodedSdkName, + 'uuid' => $this->pubnub_demo->getConfiguration()->getUuid(), ]) ->setResponseBody("{\"status\":200,\"message\":\"OK\",\"service\":\"Presence\",\"uuids\":[{\"uuid\":\"a3ffd" . "012-a3b9-478c-8705-64089f24d71e\",\"state\":{\"age\":10}}],\"occupancy\":1}"); @@ -227,13 +227,13 @@ public function testSingularChannel() */ public function testSingularChannelAndGroup() { - $hereNow = new HereNowExposed($this->pubnub); + $hereNow = new HereNowExposed($this->pubnub_demo); $hereNow->stubFor("/v2/presence/sub-key/demo/channel/game1") ->withQuery([ 'channel-group' => 'grp1', 'state' => '1', - 'uuid' => Stub::ANY, - 'pnsdk' => $this->encodedSdkName + 'pnsdk' => $this->encodedSdkName, + 'uuid' => $this->pubnub_demo->getConfiguration()->getUuid(), ]) ->setResponseBody("{\"status\":200,\"message\":\"OK\",\"payload\":{\"channels\":{}, \"total_channels\":0, " . "\"total_occupancy\":0},\"service\":\"Presence\"}"); @@ -250,12 +250,12 @@ public function testSingularChannelAndGroup() public function testIsAuthRequiredSuccess() { $this->expectNotToPerformAssertions(); - $hereNow = new HereNowExposed($this->pubnub); + $hereNow = new HereNowExposed($this->pubnub_demo); $hereNow->stubFor("/v2/presence/sub-key/demo/channel/ch1,ch2") ->withQuery([ 'state' => '1', 'pnsdk' => $this->encodedSdkName, - 'uuid' => Stub::ANY, + 'uuid' => $this->pubnub_demo->getConfiguration()->getUuid(), 'auth' => 'myKey' ]) ->setResponseBody("{\"status\":200,\"message\":\"OK\",\"payload\":{\"total_occupancy\":3,\"total_channels" @@ -263,7 +263,7 @@ public function testIsAuthRequiredSuccess() . ",\"ch2\":{\"occupancy\":2,\"uuids\":[{\"uuid\":\"user1\",\"state\":{\"age\":10}},{\"uuid\":\"user3" . "\",\"state\":{\"age\":30}}]}}},\"service\":\"Presence\"}"); - $this->pubnub->getConfiguration()->setAuthKey("myKey"); + $this->pubnub_demo->getConfiguration()->setAuthKey("myKey"); $hereNow->channels(["ch1", "ch2"])->includeState(true)->sync(); } @@ -284,7 +284,7 @@ public function testNullSubKey() ->withQuery([ 'channel-group' => 'grp1', 'state' => '1', - 'uuid' => Stub::ANY, + 'uuid' => $this->pubnub_demo->getConfiguration()->getUuid(), 'pnsdk' => $this->encodedSdkName ]) ->setResponseBody("{\"status\":200,\"message\":\"OK\",\"payload\":{\"total_occupancy\":3,\"total_channels" @@ -312,7 +312,7 @@ public function testEmptySubKey() ->withQuery([ 'state' => '1', 'pnsdk' => $this->encodedSdkName, - 'uuid' => Stub::ANY, + 'uuid' => $this->pubnub_demo->getConfiguration()->getUuid(), 'auth' => 'myKey' ]) ->setResponseBody("{\"status\":200,\"message\":\"OK\",\"payload\":{\"total_occupancy\":3,\"total_channels" @@ -339,24 +339,19 @@ public function testSuperCallTest() // phpcs:ignore PSR1.Classes.ClassDeclaration class HereNowExposed extends HereNow { - protected $transport; + protected PsrStubClient $client; - public function __construct($pubnubInstance) + public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function requestOptions() + public function stubFor(string $url): PsrStub { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } diff --git a/tests/integrational/HistoryDeleteTest.php b/tests/integrational/HistoryDeleteTest.php index 7f1b3d23..769469b0 100644 --- a/tests/integrational/HistoryDeleteTest.php +++ b/tests/integrational/HistoryDeleteTest.php @@ -2,53 +2,39 @@ namespace Tests\Integrational; -use PubNub\Exceptions\PubNubServerException; use PubNub\PubNub; use PubNub\Endpoints\HistoryDelete; use PubNub\Exceptions\PubNubValidationException; -use Tests\Helpers\Stub; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; -class TestPubNubHistoryDelete extends \PubNubTestCase +class HistoryDeleteTest extends \PubNubTestCase { - public function testMissingChannelException() + public function testMissingChannelException(): void { $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channel missing"); - $historyDelete = new HistoryDeleteExposed($this->pubnub); + $historyDelete = new HistoryDeleteExposed($this->pubnub_demo); $historyDelete->stubFor("/v2/history/sub-key/demo/channel/niceChannel") ->withQuery([ "pnsdk" => $this->encodedSdkName, - "uuid" => Stub::ANY + "uuid" => $this->pubnub_demo->getConfiguration()->getUuid(), ]) ->setResponseBody(json_encode([])); $historyDelete->sync(); } - public function testChannelIsEmptyException() + public function testChannelIsEmptyException(): void { $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channel missing"); - - $historyDelete = new HistoryDeleteExposed($this->pubnub); - - $historyDelete->channel("")->sync(); + $this->pubnub->deleteMessages()->channel("")->sync(); } - // This test will require a key with specific permissions assigned - // public function testNotPermitted() - // { - // $channel = "history-delete-php-ch"; - // $this->expectException(PubNubServerException::class); - - // $this->pubnub_pam->getConfiguration()->setSecretKey(null); - // $this->pubnub_pam->deleteMessages()->channel($channel)->start(123)->end(456)->sync(); - // } - - public function testSuperCallTest() + public function testSuperCallTest(): void { $this->expectNotToPerformAssertions(); $this->pubnub_pam->deleteMessages() @@ -60,24 +46,19 @@ public function testSuperCallTest() // phpcs:ignore PSR1.Classes.ClassDeclaration class HistoryDeleteExposed extends HistoryDelete { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function requestOptions() + public function stubFor(string $url): PsrStub { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } diff --git a/tests/integrational/HistoryTest.php b/tests/integrational/HistoryTest.php index ac4b2172..ec57663b 100644 --- a/tests/integrational/HistoryTest.php +++ b/tests/integrational/HistoryTest.php @@ -3,13 +3,12 @@ namespace Tests\Integrational; use PubNub\Exceptions\PubNubResponseParsingException; -use PubNub\Exceptions\PubNubServerException; use PubNub\Models\Consumer\History\PNHistoryResult; use PubNub\PubNub; use PubNub\Endpoints\History; use PubNub\Exceptions\PubNubValidationException; -use Tests\Helpers\Stub; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class HistoryTest extends \PubNubTestCase { @@ -19,7 +18,7 @@ class HistoryTest extends \PubNubTestCase */ public function testSuccess() { - $history = new HistoryExposed($this->pubnub); + $history = new HistoryExposed($this->pubnub_demo); $testArray = []; $historyItems = []; @@ -52,8 +51,8 @@ public function testSuccess() ->withQuery([ "count" => "100", "include_token" => "true", - "pnsdk" => $this->encodedSdkName, - "uuid" => Stub::ANY + "pnsdk" => $this->pubnub_demo->getSdkFullName(), + "uuid" => $this->pubnub_demo->getConfiguration()->getUuid() ]) ->setResponseBody(json_encode($testArray)); @@ -77,8 +76,8 @@ public function testSuccess() public function testAuthSuccess() { $this->expectNotToPerformAssertions(); - $this->pubnub->getConfiguration()->setAuthKey("blah"); - $history = new HistoryExposed($this->pubnub); + $this->pubnub_demo->getConfiguration()->setAuthKey("blah"); + $history = new HistoryExposed($this->pubnub_demo); $testArray = []; $historyItems = []; @@ -111,9 +110,9 @@ public function testAuthSuccess() ->withQuery([ "count" => "100", "include_token" => "true", - "pnsdk" => $this->encodedSdkName, - "uuid" => Stub::ANY, - "auth" => "auth" + "pnsdk" => $this->pubnub_demo->getSdkFullName(), + "uuid" => $this->pubnub_demo->getConfiguration()->getUuid(), + "auth" => "blah" ]) ->setResponseBody(json_encode($testArray)); @@ -131,6 +130,9 @@ public function testEncryptedSuccess() $config = $this->config->clone(); $config->setUseRandomIV(false); $config->setCipherKey("cipherKey"); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUuid("demo"); $pubnub = new PubNub($config); $history = new HistoryExposed($pubnub); @@ -138,8 +140,8 @@ public function testEncryptedSuccess() ->withQuery([ "count" => "100", "include_token" => "true", - "pnsdk" => $this->encodedSdkName, - "uuid" => Stub::ANY + "pnsdk" => $pubnub->getSdkFullName(), + "uuid" => $pubnub->getConfiguration()->getUuid(), ]) ->setResponseBody("[[{\"message\":\"zFJeF9BVABL80GUiQEBjLg==\"," . "\"timetoken\":\"14649369736959785\"}," @@ -166,6 +168,9 @@ public function testEncryptedWithPNOtherSuccess() $config = $this->config->clone(); $config->setUseRandomIV(false); $config->setCipherKey("hello"); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUuid("demo"); $pubnub = new PubNub($config); $history = new HistoryExposed($pubnub); @@ -173,8 +178,8 @@ public function testEncryptedWithPNOtherSuccess() ->withQuery([ "count" => "100", "include_token" => "false", - "pnsdk" => $this->encodedSdkName, - "uuid" => Stub::ANY + "pnsdk" => $pubnub->getSdkFullName(), + "uuid" => $pubnub->getConfiguration()->getUuid(), ]) ->setResponseBody("[[{\"pn_other\":\"6QoqmS9CnB3W9+I4mhmL7w==\"}],14606134331557852,14606134485013970]"); @@ -188,7 +193,7 @@ public function testEncryptedWithPNOtherSuccess() public function testSuccessWithoutTimeToken() { - $history = new HistoryExposed($this->pubnub); + $history = new HistoryExposed($this->pubnub_demo); $testArray = []; $historyItems = []; @@ -211,8 +216,8 @@ public function testSuccessWithoutTimeToken() $history->stubFor("/v2/history/sub-key/demo/channel/niceChannel") ->withQuery([ "count" => "100", - "pnsdk" => $this->encodedSdkName, - "uuid" => Stub::ANY + "pnsdk" => $this->pubnub_demo->getSdkFullName(), + "uuid" => $this->pubnub_demo->getConfiguration()->getUuid() ]) ->setResponseBody(json_encode($testArray)); @@ -237,12 +242,12 @@ public function testMissingChannel() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channel missing"); - $history = new HistoryExposed($this->pubnub); + $history = new HistoryExposed($this->pubnub_demo); $history->stubFor("/v2/history/sub-key/demo/channel/niceChannel") ->withQuery([ - "pnsdk" => $this->encodedSdkName, - "uuid" => Stub::ANY + "pnsdk" => $this->pubnub_demo->getSdkFullName(), + "uuid" => $this->pubnub_demo->getConfiguration()->getUuid(), ]) ->setResponseBody(json_encode([])); @@ -254,12 +259,12 @@ public function testChannelIsEmpty() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channel missing"); - $history = new HistoryExposed($this->pubnub); + $history = new HistoryExposed($this->pubnub_demo); $history->stubFor("/v2/history/sub-key/demo/channel/niceChannel") ->withQuery([ - "pnsdk" => $this->encodedSdkName, - "uuid" => Stub::ANY + "pnsdk" => $this->pubnub_demo->getSdkFullName(), + "uuid" => $this->pubnub_demo->getConfiguration()->getUuid() ]) ->setResponseBody(json_encode([])); @@ -268,7 +273,7 @@ public function testChannelIsEmpty() public function testCountReverseStartEndSuccess() { - $history = new HistoryExposed($this->pubnub); + $history = new HistoryExposed($this->pubnub_demo); $testArray = []; $historyItems = []; @@ -304,8 +309,8 @@ public function testCountReverseStartEndSuccess() "count" => "5", "reverse" => "true", "include_token" => "true", - "pnsdk" => $this->encodedSdkName, - "uuid" => Stub::ANY + "pnsdk" => $this->pubnub_demo->getSdkFullName(), + "uuid" => $this->pubnub_demo->getConfiguration()->getUuid() ]) ->setResponseBody(json_encode($testArray)); @@ -337,9 +342,9 @@ public function testProcessMessageError() $this->expectException(PubNubResponseParsingException::class); $this->expectExceptionMessage("Decryption error: message is not a string"); - $history = new HistoryExposed($this->pubnub); + $history = new HistoryExposed($this->pubnub_demo); - $this->pubnub->getConfiguration()->setCipherKey("Test"); + $this->pubnub_demo->getConfiguration()->setCipherKey("Test"); $testArray = []; $historyItems = []; @@ -375,8 +380,8 @@ public function testProcessMessageError() "count" => "5", "reverse" => "true", "include_token" => "true", - "pnsdk" => $this->encodedSdkName, - "uuid" => Stub::ANY + "pnsdk" => $this->pubnub_demo->getSdkFullName(), + "uuid" => $this->pubnub_demo->getConfiguration()->getUuid() ]) ->setResponseBody(json_encode($testArray)); @@ -446,24 +451,19 @@ public function testSuperCallTest() // phpcs:ignore PSR1.Classes.ClassDeclaration class HistoryExposed extends History { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function requestOptions() + public function stubFor(string $url): PsrStub { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } diff --git a/tests/integrational/ListChannelsInChannelGroupTest.php b/tests/integrational/ListChannelsInChannelGroupTest.php index 5e8bbdcb..44317088 100644 --- a/tests/integrational/ListChannelsInChannelGroupTest.php +++ b/tests/integrational/ListChannelsInChannelGroupTest.php @@ -7,13 +7,14 @@ use PubNub\Exceptions\PubNubValidationException; use PubNub\PubNub; use PubNubTestCase; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class ListChannelsInChannelGroupTest extends PubNubTestCase { public function testSuccess() { - $listChannelsInChannelGroup = new ListChannelsInChannelGroupExposed($this->pubnub); + $listChannelsInChannelGroup = new ListChannelsInChannelGroupExposed($this->pubnub_demo); $listChannelsInChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/groupA") ->withQuery([ @@ -23,7 +24,7 @@ public function testSuccess() ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {\"channels\": [\"a\",\"b\"]}, " . "\"service\": \"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); $response = $listChannelsInChannelGroup->channelGroup("groupA")->sync(); @@ -35,7 +36,7 @@ public function testGroupMissing() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channel group missing"); - $listChannelsInChannelGroup = new ListChannelsInChannelGroupExposed($this->pubnub); + $listChannelsInChannelGroup = new ListChannelsInChannelGroupExposed($this->pubnub_demo); $listChannelsInChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/groupA") ->withQuery([ @@ -45,7 +46,7 @@ public function testGroupMissing() ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {\"channels\": [\"a\",\"b\"]}, " . "\"service\": \"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); $listChannelsInChannelGroup->sync(); } @@ -55,7 +56,7 @@ public function testEmptyGroup() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channel group missing"); - $listChannelsInChannelGroup = new ListChannelsInChannelGroupExposed($this->pubnub); + $listChannelsInChannelGroup = new ListChannelsInChannelGroupExposed($this->pubnub_demo); $listChannelsInChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/groupA") ->withQuery([ @@ -65,7 +66,7 @@ public function testEmptyGroup() ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {\"channels\": [\"a\",\"b\"]}, " . "\"service\": \"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); $listChannelsInChannelGroup->channelGroup("")->sync(); } @@ -75,7 +76,7 @@ public function testNullPayload() $this->expectException(PubNubResponseParsingException::class); $this->expectExceptionMessage("Unable to parse server response: No payload found in response"); - $listChannelsInChannelGroup = new ListChannelsInChannelGroupExposed($this->pubnub); + $listChannelsInChannelGroup = new ListChannelsInChannelGroupExposed($this->pubnub_demo); $listChannelsInChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/groupA") ->withQuery([ @@ -84,7 +85,7 @@ public function testNullPayload() ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"service\": \"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); $listChannelsInChannelGroup->channelGroup("groupA")->sync(); } @@ -93,16 +94,16 @@ public function testNullBody() { $this->expectException(PubNubResponseParsingException::class); - $listChannelsInChannelGroup = new ListChannelsInChannelGroupExposed($this->pubnub); + $listChannelsInChannelGroup = new ListChannelsInChannelGroupExposed($this->pubnub_demo); $listChannelsInChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/groupA") ->withQuery([ "pnsdk" => $this->encodedSdkName, "uuid" => "myUUID" ]) - ->setResponseBody(""); + ->setResponseBody(json_encode("")); - $this->pubnub->getConfiguration()->setUuid("myUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); $listChannelsInChannelGroup->channelGroup("groupA")->sync(); } @@ -110,7 +111,7 @@ public function testNullBody() public function testIsAuthRequiredSuccess() { $this->expectNotToPerformAssertions(); - $listChannelsInChannelGroup = new ListChannelsInChannelGroupExposed($this->pubnub); + $listChannelsInChannelGroup = new ListChannelsInChannelGroupExposed($this->pubnub_demo); $listChannelsInChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/groupA") ->withQuery([ @@ -121,7 +122,7 @@ public function testIsAuthRequiredSuccess() ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {\"channels\": [\"a\",\"b\"]}, " . "\"service\": \"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID")->setAuthKey("myKey"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID")->setAuthKey("myKey"); $listChannelsInChannelGroup->channelGroup("groupA")->sync(); } @@ -130,24 +131,19 @@ public function testIsAuthRequiredSuccess() // phpcs:ignore PSR1.Classes.ClassDeclaration class ListChannelsInChannelGroupExposed extends ListChannelsInChannelGroup { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function requestOptions() + public function stubFor(string $url): PsrStub { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } diff --git a/tests/integrational/ListPushProvisionsTest.php b/tests/integrational/ListPushProvisionsTest.php index 49164cab..2bd332c9 100644 --- a/tests/integrational/ListPushProvisionsTest.php +++ b/tests/integrational/ListPushProvisionsTest.php @@ -6,19 +6,20 @@ use PubNub\Enums\PNPushType; use PubNub\Exceptions\PubNubValidationException; use PubNub\PubNub; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class ListPushProvisionsTest extends \PubNubTestCase { public function testAppleSuccess() { - $list = new ListPushProvisionsExposed($this->pubnub); + $list = new ListPushProvisionsExposed($this->pubnub_demo); $list->stubFor("/v1/push/sub-key/demo/devices/coolDevice") ->withQuery([ "type" => "apns", "pnsdk" => $this->encodedSdkName, - "uuid" => getenv("UUID_MOCK") + "uuid" => $this->pubnub_demo->getConfiguration()->getUuid(), ]) ->setResponseBody("[\"ch1\", \"ch2\", \"ch3\"]"); @@ -31,13 +32,13 @@ public function testAppleSuccess() public function testFCMSuccess() { - $list = new ListPushProvisionsExposed($this->pubnub); + $list = new ListPushProvisionsExposed($this->pubnub_demo); $list->stubFor("/v1/push/sub-key/demo/devices/coolDevice") ->withQuery([ "type" => "gcm", "pnsdk" => $this->encodedSdkName, - "uuid" => getenv("UUID_MOCK") + "uuid" => $this->pubnub_demo->getConfiguration()->getUuid(), ]) ->setResponseBody("[\"ch1\", \"ch2\", \"ch3\"]"); @@ -50,13 +51,13 @@ public function testFCMSuccess() public function testMicrosoftSuccess() { - $list = new ListPushProvisionsExposed($this->pubnub); + $list = new ListPushProvisionsExposed($this->pubnub_demo); $list->stubFor("/v1/push/sub-key/demo/devices/coolDevice") ->withQuery([ "type" => "mpns", "pnsdk" => $this->encodedSdkName, - "uuid" => getenv("UUID_MOCK") + "uuid" => $this->pubnub_demo->getConfiguration()->getUuid(), ]) ->setResponseBody("[\"ch1\", \"ch2\", \"ch3\"]"); @@ -71,15 +72,19 @@ public function testIsAuthRequiredSuccess() { $config = $this->config->clone(); $config->setAuthKey("myKey"); + $config->setUseRandomIV(false); + $config->setCipherKey("cipherKey"); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $pubnub = new PubNub($config); $list = new ListPushProvisionsExposed($pubnub); $list->stubFor("/v1/push/sub-key/demo/devices/coolDevice") ->withQuery([ - "auth" => "myKey", "type" => "mpns", "pnsdk" => $this->encodedSdkName, - "uuid" => getenv("UUID_MOCK") + "uuid" => $this->pubnub_demo->getConfiguration()->getUuid(), + "auth" => "myKey", ]) ->setResponseBody("[\"ch1\", \"ch2\", \"ch3\"]"); @@ -123,7 +128,7 @@ public function testNullPushType() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Push Type is missing"); - $list = new ListPushProvisionsExposed($this->pubnub); + $list = new ListPushProvisionsExposed($this->pubnub_demo); $list->deviceId("coolDevice") ->sync(); @@ -134,7 +139,7 @@ public function testNullDeviceId() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Device ID is missing for push operation"); - $list = new ListPushProvisionsExposed($this->pubnub); + $list = new ListPushProvisionsExposed($this->pubnub_demo); $list->pushType(PNPushType::MPNS) ->sync(); @@ -145,52 +150,30 @@ public function testEmptyDeviceIdRemoveAll() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Device ID is missing for push operation"); - $list = new ListPushProvisionsExposed($this->pubnub); + $list = new ListPushProvisionsExposed($this->pubnub_demo); $list->deviceId("") ->pushType(PNPushType::MPNS) ->sync(); } - - public function superCallTest() - { - $this->pubnub_pam->listPushProvisions() - ->deviceId(static::SPECIAL_CHARACTERS) - ->sync(); - } } // phpcs:ignore PSR1.Classes.ClassDeclaration class ListPushProvisionsExposed extends ListPushProvisions { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); - } - - public function buildParams() - { - return parent::buildParams(); - } - - public function buildPath() - { - return parent::buildPath(); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function requestOptions() + public function stubFor(string $url): PsrStub { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } diff --git a/tests/integrational/ModifyPushChannelsForDeviceTest.php b/tests/integrational/ModifyPushChannelsForDeviceTest.php index 46e4f48f..96822c52 100644 --- a/tests/integrational/ModifyPushChannelsForDeviceTest.php +++ b/tests/integrational/ModifyPushChannelsForDeviceTest.php @@ -7,14 +7,18 @@ use PubNub\Endpoints\Push\RemoveDeviceFromPush; use PubNub\Enums\PNPushType; use PubNub\Exceptions\PubNubValidationException; +use PubNub\PNConfiguration; use PubNub\PubNub; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class ModifyPushChannelsForDeviceTest extends \PubNubTestCase { public function testListChannelGroupAPNS() { - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); @@ -37,7 +41,9 @@ public function testListChannelGroupAPNS() public function testFCMSuccessRemoveAll() { - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); @@ -60,7 +66,9 @@ public function testFCMSuccessRemoveAll() public function testMicrosoftSuccessRemoveAll() { - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); @@ -83,7 +91,9 @@ public function testMicrosoftSuccessRemoveAll() public function testIsAuthRequiredSuccessRemoveAll() { - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $config->setAuthKey('myKey'); $pubnub = new PubNub($config); @@ -92,10 +102,10 @@ public function testIsAuthRequiredSuccessRemoveAll() $listRemove->stubFor("/v1/push/sub-key/demo/devices/coolDevice/remove") ->withQuery([ - "auth" => "myKey", "type" => "mpns", "pnsdk" => $this->encodedSdkName, - "uuid" => "sampleUUID" + "uuid" => "sampleUUID", + "auth" => "myKey", ]) ->setResponseBody("[1, \"Modified Channels\"]"); @@ -109,7 +119,9 @@ public function testIsAuthRequiredSuccessRemoveAll() public function testNullSubscribeKeyRemoveAll() { $this->expectException(\TypeError::class); - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $config->setSubscribeKey(null); $pubnub = new PubNub($config); @@ -126,7 +138,9 @@ public function testEmptySubscribeKeyRemoveAll() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Subscribe Key not configured"); - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $config->setSubscribeKey(""); $pubnub = new PubNub($config); @@ -143,7 +157,9 @@ public function testNullPushTypeRemoveAll() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Push Type is missing"); - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); @@ -158,7 +174,9 @@ public function testNullDeviceIdRemoveAll() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Device ID is missing for push operation"); - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); @@ -173,7 +191,9 @@ public function testEmptyDeviceIdRemoveAll() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Device ID is missing for push operation"); - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); @@ -186,7 +206,9 @@ public function testEmptyDeviceIdRemoveAll() public function testAddAppleSuccess() { - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); @@ -211,7 +233,9 @@ public function testAddAppleSuccess() public function testAddFCMSuccessSync() { - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); @@ -236,7 +260,9 @@ public function testAddFCMSuccessSync() public function testAddMicrosoftSuccessSync() { - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); @@ -261,7 +287,9 @@ public function testAddMicrosoftSuccessSync() public function testIsAuthRequiredSuccessAdd() { - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $config->setAuthKey("myKey"); $pubnub = new PubNub($config); @@ -271,10 +299,10 @@ public function testIsAuthRequiredSuccessAdd() $listAdd->stubFor("/v1/push/sub-key/demo/devices/coolDevice") ->withQuery([ "add" => "ch1,ch2,ch3", - "auth" => "myKey", "type" => "mpns", "pnsdk" => $this->encodedSdkName, - "uuid" => "sampleUUID" + "uuid" => "sampleUUID", + "auth" => "myKey", ]) ->setResponseBody("[1, \"Modified Channels\"]"); @@ -290,7 +318,9 @@ public function testNullSubscribeKeyAdd() { $this->expectException(\TypeError::class); - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $config->setSubscribeKey(null); $pubnub = new PubNub($config); @@ -307,7 +337,9 @@ public function testEmptySubscribeKeyAdd() { $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Subscribe Key not configured"); - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $config->setSubscribeKey(''); $pubnub = new PubNub($config); @@ -324,7 +356,9 @@ public function testNullPushTypeAdd() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Push Type is missing"); - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); $listAdd = new AddChannelsToPushExposed($pubnub); @@ -339,7 +373,9 @@ public function testNullDeviceIdAdd() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Device ID is missing for push operation"); - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); $listAdd = new AddChannelsToPushExposed($pubnub); @@ -354,7 +390,9 @@ public function testEmptyDeviceIdAdd() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Device ID is missing for push operation"); - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); $listAdd = new AddChannelsToPushExposed($pubnub); @@ -370,7 +408,9 @@ public function testMissingChannelsAdd() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channel missing"); - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); $listAdd = new AddChannelsToPushExposed($pubnub); @@ -382,7 +422,9 @@ public function testMissingChannelsAdd() public function testAppleSuccessRemove() { - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); $remove = new RemovePushNotificationsFromChannelsExposed($pubnub); @@ -406,7 +448,9 @@ public function testAppleSuccessRemove() public function testFCMSuccessRemove() { - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); $remove = new RemovePushNotificationsFromChannelsExposed($pubnub); @@ -430,7 +474,9 @@ public function testFCMSuccessRemove() public function testMicrosoftSuccessRemove() { - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); $remove = new RemovePushNotificationsFromChannelsExposed($pubnub); @@ -454,7 +500,9 @@ public function testMicrosoftSuccessRemove() public function testIsAuthRequiredSuccessRemove() { - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $config->setAuthKey("myKey"); $pubnub = new PubNub($config); @@ -462,11 +510,11 @@ public function testIsAuthRequiredSuccessRemove() $remove->stubFor("/v1/push/sub-key/demo/devices/coolDevice") ->withQuery([ - "auth" => "myKey", "remove" => "ch1,ch2,ch3", "type" => "mpns", "pnsdk" => $this->encodedSdkName, - "uuid" => "sampleUUID" + "uuid" => "sampleUUID", + "auth" => "myKey", ]) ->setResponseBody("[1, \"Modified Channels\"]"); @@ -482,7 +530,9 @@ public function testNullSubscribeKeyRemove() { $this->expectException(\TypeError::class); - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $config->setSubscribeKey(null); $pubnub = new PubNub($config); @@ -499,7 +549,9 @@ public function testEmptySubscribeKeyRemove() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Subscribe Key not configured"); - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $config->setSubscribeKey(''); $pubnub = new PubNub($config); @@ -545,7 +597,9 @@ public function testEmptyDeviceId() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Device ID is missing for push operation"); - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); $remove = new RemovePushNotificationsFromChannelsExposed($pubnub); @@ -561,7 +615,9 @@ public function testMissingChannels() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channel missing"); - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); $remove = new RemovePushNotificationsFromChannelsExposed($pubnub); @@ -583,104 +639,59 @@ public function superCallTest() // phpcs:ignore PSR1.Classes.ClassDeclaration class RemoveChannelsFromPushTestExposed extends RemoveDeviceFromPush { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function stubFor($url) + public function stubFor(string $url): PsrStub { - return $this->transport->stubFor($url); - } - - public function buildParams() - { - return parent::buildParams(); - } - - public function buildPath() - { - return parent::buildPath(); - } - - public function requestOptions() - { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } // phpcs:ignore PSR1.Classes.ClassDeclaration class AddChannelsToPushExposed extends AddChannelsToPush { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function stubFor($url) + public function stubFor(string $url): PsrStub { - return $this->transport->stubFor($url); - } - - public function buildParams() - { - return parent::buildParams(); - } - - public function buildPath() - { - return parent::buildPath(); - } - - public function requestOptions() - { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } // phpcs:ignore PSR1.Classes.ClassDeclaration class RemovePushNotificationsFromChannelsExposed extends RemoveChannelsFromPush { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); - } - - public function buildParams() - { - return parent::buildParams(); - } - - public function buildPath() - { - return parent::buildPath(); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function requestOptions() + public function stubFor(string $url): PsrStub { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } diff --git a/tests/integrational/PamTest.php b/tests/integrational/PamTest.php index 0c4f5859..ef9f9c9f 100644 --- a/tests/integrational/PamTest.php +++ b/tests/integrational/PamTest.php @@ -19,7 +19,6 @@ class PamTest extends \PubNubTestCase public function testGlobalLevel() { $response = $this->pubnub_pam->grant()->read(true)->write(true)->sync(); - var_dump($response); $this->assertInstanceOf(PNAccessManagerAbstractResult::class, $response); $this->assertCount(0, $response->getChannels()); @@ -294,21 +293,21 @@ public function testMultipleUuid() ->join(false) ->sync(); - $this->assertInstanceOf(PNAccessManagerAbstractResult::class, $response); - $this->assertEquals("uuid", $response->getLevel()); - $this->assertTrue($response->getUsers()[$targetUuid1]->getAuthKeys()[$auth]->isReadEnabled()); - $this->assertTrue($response->getUsers()[$targetUuid2]->getAuthKeys()[$auth]->isReadEnabled()); - $this->assertTrue($response->getUsers()[$targetUuid1]->getAuthKeys()[$auth]->isWriteEnabled()); - $this->assertTrue($response->getUsers()[$targetUuid2]->getAuthKeys()[$auth]->isWriteEnabled()); - $this->assertTrue($response->getUsers()[$targetUuid1]->getAuthKeys()[$auth]->isManageEnabled()); - $this->assertTrue($response->getUsers()[$targetUuid2]->getAuthKeys()[$auth]->isManageEnabled()); - $this->assertTrue($response->getUsers()[$targetUuid1]->getAuthKeys()[$auth]->isDeleteEnabled()); - $this->assertTrue($response->getUsers()[$targetUuid2]->getAuthKeys()[$auth]->isDeleteEnabled()); - $this->assertTrue($response->getUsers()[$targetUuid1]->getAuthKeys()[$auth]->isGetEnabled()); - $this->assertTrue($response->getUsers()[$targetUuid2]->getAuthKeys()[$auth]->isGetEnabled()); - $this->assertTrue($response->getUsers()[$targetUuid1]->getAuthKeys()[$auth]->isUpdateEnabled()); - $this->assertTrue($response->getUsers()[$targetUuid2]->getAuthKeys()[$auth]->isUpdateEnabled()); - $this->assertFalse($response->getUsers()[$targetUuid1]->getAuthKeys()[$auth]->isJoinEnabled()); - $this->assertFalse($response->getUsers()[$targetUuid2]->getAuthKeys()[$auth]->isJoinEnabled()); - } + $this->assertInstanceOf(PNAccessManagerAbstractResult::class, $response); + $this->assertEquals("uuid", $response->getLevel()); + $this->assertTrue($response->getUsers()[$targetUuid1]->getAuthKeys()[$auth]->isReadEnabled()); + $this->assertTrue($response->getUsers()[$targetUuid2]->getAuthKeys()[$auth]->isReadEnabled()); + $this->assertTrue($response->getUsers()[$targetUuid1]->getAuthKeys()[$auth]->isWriteEnabled()); + $this->assertTrue($response->getUsers()[$targetUuid2]->getAuthKeys()[$auth]->isWriteEnabled()); + $this->assertTrue($response->getUsers()[$targetUuid1]->getAuthKeys()[$auth]->isManageEnabled()); + $this->assertTrue($response->getUsers()[$targetUuid2]->getAuthKeys()[$auth]->isManageEnabled()); + $this->assertTrue($response->getUsers()[$targetUuid1]->getAuthKeys()[$auth]->isDeleteEnabled()); + $this->assertTrue($response->getUsers()[$targetUuid2]->getAuthKeys()[$auth]->isDeleteEnabled()); + $this->assertTrue($response->getUsers()[$targetUuid1]->getAuthKeys()[$auth]->isGetEnabled()); + $this->assertTrue($response->getUsers()[$targetUuid2]->getAuthKeys()[$auth]->isGetEnabled()); + $this->assertTrue($response->getUsers()[$targetUuid1]->getAuthKeys()[$auth]->isUpdateEnabled()); + $this->assertTrue($response->getUsers()[$targetUuid2]->getAuthKeys()[$auth]->isUpdateEnabled()); + $this->assertFalse($response->getUsers()[$targetUuid1]->getAuthKeys()[$auth]->isJoinEnabled()); + $this->assertFalse($response->getUsers()[$targetUuid2]->getAuthKeys()[$auth]->isJoinEnabled()); + } } diff --git a/tests/integrational/RemoveChannelFromChannelGroupEndpointTest.php b/tests/integrational/RemoveChannelFromChannelGroupEndpointTest.php index 92987507..e35688de 100644 --- a/tests/integrational/RemoveChannelFromChannelGroupEndpointTest.php +++ b/tests/integrational/RemoveChannelFromChannelGroupEndpointTest.php @@ -6,23 +6,24 @@ use PubNub\Exceptions\PubNubValidationException; use PubNub\PubNub; use PubNubTestCase; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class RemoveChannelFromChannelGroupEndpointTest extends PubNubTestCase { public function testSuccess() { - $removeChannelFromChannelGroup = new RemoveChannelFromChannelGroupExposed($this->pubnub); + $removeChannelFromChannelGroup = new RemoveChannelFromChannelGroupExposed($this->pubnub_demo); $removeChannelFromChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/groupA") ->withQuery([ "pnsdk" => $this->encodedSdkName, "uuid" => "myUUID", - "remove" => "ch1,ch2" + "remove" => "ch1,ch2", ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {},\"service\": \"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); $response = $removeChannelFromChannelGroup->channelGroup("groupA")->channels(["ch1", "ch2"])->sync(); @@ -34,16 +35,16 @@ public function testGroupMissing() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channel group missing"); - $removeChannelFromChannelGroup = new RemoveChannelFromChannelGroupExposed($this->pubnub); + $removeChannelFromChannelGroup = new RemoveChannelFromChannelGroupExposed($this->pubnub_demo); $removeChannelFromChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/groupA") ->withQuery([ "pnsdk" => $this->encodedSdkName, - "uuid" => "myUUID" + "uuid" => "myUUID", ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {},\"service\": \"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); $removeChannelFromChannelGroup->channels(["ch1" ,"ch2"])->sync(); } @@ -53,7 +54,7 @@ public function testEmptyGroup() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channels missing"); - $removeChannelFromChannelGroup = new RemoveChannelFromChannelGroupExposed($this->pubnub); + $removeChannelFromChannelGroup = new RemoveChannelFromChannelGroupExposed($this->pubnub_demo); $removeChannelFromChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/groupA") ->withQuery([ @@ -62,7 +63,7 @@ public function testEmptyGroup() ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {},\"service\": \"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); $removeChannelFromChannelGroup->channelGroup("groupA")->sync(); } @@ -70,18 +71,18 @@ public function testEmptyGroup() public function testIsAuthRequiredSuccess() { $this->expectNotToPerformAssertions(); - $removeChannelFromChannelGroup = new RemoveChannelFromChannelGroupExposed($this->pubnub); + $removeChannelFromChannelGroup = new RemoveChannelFromChannelGroupExposed($this->pubnub_demo); $removeChannelFromChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/groupA") ->withQuery([ "pnsdk" => $this->encodedSdkName, "uuid" => "myUUID", "auth" => "myKey", - "remove" => "ch1,ch2" + "remove" => "ch1,ch2", ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {},\"service\": \"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID")->setAuthKey("myKey"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID")->setAuthKey("myKey"); $removeChannelFromChannelGroup->channelGroup("groupA")->channels(["ch1", "ch2"])->sync(); } @@ -90,24 +91,19 @@ public function testIsAuthRequiredSuccess() // phpcs:ignore PSR1.Classes.ClassDeclaration class RemoveChannelFromChannelGroupExposed extends RemoveChannelFromChannelGroup { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function requestOptions() + public function stubFor(string $url): PsrStub { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } diff --git a/tests/integrational/RemoveChannelGroupEndpointTest.php b/tests/integrational/RemoveChannelGroupEndpointTest.php index 3ed8babc..6903924e 100644 --- a/tests/integrational/RemoveChannelGroupEndpointTest.php +++ b/tests/integrational/RemoveChannelGroupEndpointTest.php @@ -6,22 +6,23 @@ use PubNub\Exceptions\PubNubValidationException; use PubNub\PubNub; use PubNubTestCase; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class RemoveChannelGroupEndpointTest extends PubNubTestCase { public function testSuccess() { - $removeChannelGroup = new RemoveChannelGroupExposed($this->pubnub); + $removeChannelGroup = new RemoveChannelGroupExposed($this->pubnub_demo); $removeChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/groupA/remove") ->withQuery([ "pnsdk" => $this->encodedSdkName, - "uuid" => "myUUID" + "uuid" => "myUUID", ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {},\"service\": \"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); $response = $removeChannelGroup->channelGroup("groupA")->sync(); @@ -33,16 +34,16 @@ public function testGroupMissing() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channel group missing"); - $removeChannelGroup = new RemoveChannelGroupExposed($this->pubnub); + $removeChannelGroup = new RemoveChannelGroupExposed($this->pubnub_demo); $removeChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/groupA/remove") ->withQuery([ "pnsdk" => $this->encodedSdkName, - "uuid" => "myUUID" + "uuid" => "myUUID", ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {},\"service\": \"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); $removeChannelGroup->sync(); } @@ -52,16 +53,16 @@ public function testEmptyGroup() $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Channel group missing"); - $removeChannelGroup = new RemoveChannelGroupExposed($this->pubnub); + $removeChannelGroup = new RemoveChannelGroupExposed($this->pubnub_demo); $removeChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/groupA/remove") ->withQuery([ "pnsdk" => $this->encodedSdkName, - "uuid" => "myUUID" + "uuid" => "myUUID", ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {},\"service\": \"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); $removeChannelGroup->channelGroup("")->sync(); } @@ -69,17 +70,17 @@ public function testEmptyGroup() public function testIsAuthRequiredSuccess() { $this->expectNotToPerformAssertions(); - $removeChannelGroup = new RemoveChannelGroupExposed($this->pubnub); + $removeChannelGroup = new RemoveChannelGroupExposed($this->pubnub_demo); $removeChannelGroup->stubFor("/v1/channel-registration/sub-key/demo/channel-group/groupA/remove") ->withQuery([ "pnsdk" => $this->encodedSdkName, "uuid" => "myUUID", - "auth" => "myKey" + "auth" => "myKey", ]) ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {},\"service\": \"ChannelGroups\"}"); - $this->pubnub->getConfiguration()->setUuid("myUUID")->setAuthKey("myKey"); + $this->pubnub_demo->getConfiguration()->setUuid("myUUID")->setAuthKey("myKey"); $removeChannelGroup->channelGroup("groupA")->sync(); } @@ -95,24 +96,19 @@ public function superCallTest() // phpcs:ignore PSR1.Classes.ClassDeclaration class RemoveChannelGroupExposed extends RemoveChannelGroup { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function requestOptions() + public function stubFor(string $url): PsrStub { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } diff --git a/tests/integrational/SetStateTest.php b/tests/integrational/SetStateTest.php index 6978817b..4d87fe11 100644 --- a/tests/integrational/SetStateTest.php +++ b/tests/integrational/SetStateTest.php @@ -6,15 +6,20 @@ use PubNub\PubNub; use PubNub\Exceptions\PubNubException; use PubNub\Exceptions\PubNubServerException; +use PubNub\Exceptions\PubNubValidationException; +use PubNub\PNConfiguration; use PubNub\PubNubUtil; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class SetStateTest extends \PubNubTestCase { public function testApplyStateForChannel() { - $config = $this->config->clone(); - $config->setUuid("myUUID"); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUserId("myUserId"); $pubnub = new PubNub($config); $setState = new SetStateExposed($pubnub); @@ -22,11 +27,11 @@ public function testApplyStateForChannel() "age" => 20 ]; - $setState->stubFor("/v2/presence/sub-key/demo/channel/testChannel/uuid/myUUID/data") + $setState->stubFor("/v2/presence/sub-key/demo/channel/testChannel/uuid/myUserId/data") ->withQuery([ "state" => "%7B%22age%22%3A20%7D", "pnsdk" => $this->encodedSdkName, - "uuid" => "myUUID" + "uuid" => "myUserId" ]) ->setResponseBody("{ \"status\": 200, \"message\": \"OK\", \"payload\": { \"age\" : 20, " . "\"status\" : \"online\" }, \"service\": \"Presence\"}"); @@ -39,8 +44,10 @@ public function testApplyStateForChannel() public function testApplyStateForSomebodyElseChannel() { - $config = $this->config->clone(); - $config->setUuid("someoneElseUUID"); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUserId("someoneElseUUID"); $pubnub = new PubNub($config); $setState = new SetStateExposed($pubnub); @@ -50,9 +57,9 @@ public function testApplyStateForSomebodyElseChannel() $setState->stubFor("/v2/presence/sub-key/demo/channel/testChannel/uuid/someoneElseUUID/data") ->withQuery([ - "uuid" => "someoneElseUUID", "state" => "%7B%22age%22%3A20%7D", - "pnsdk" => $this->encodedSdkName + "pnsdk" => $this->encodedSdkName, + "uuid" => "someoneElseUUID", ]) ->setResponseBody("{ \"status\": 200, \"message\": \"OK\", \"payload\": { \"age\" : 20, \"status\" : " . "\"online\" }, \"service\": \"Presence\"}"); @@ -65,8 +72,10 @@ public function testApplyStateForSomebodyElseChannel() public function testApplyStateForChannelsSync() { - $config = $this->config->clone(); - $config->setUuid("myUUID"); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUserId("myUserId"); $pubnub = new PubNub($config); $setState = new SetStateExposed($pubnub); @@ -74,11 +83,11 @@ public function testApplyStateForChannelsSync() "age" => 20 ]; - $setState->stubFor("/v2/presence/sub-key/demo/channel/testChannel,testChannel2/uuid/myUUID/data") + $setState->stubFor("/v2/presence/sub-key/demo/channel/testChannel,testChannel2/uuid/myUserId/data") ->withQuery([ - "uuid" => "myUUID", "state" => "%7B%22age%22%3A20%7D", - "pnsdk" => $this->encodedSdkName + "pnsdk" => $this->encodedSdkName, + "uuid" => "myUserId", ]) ->setResponseBody("{ \"status\": 200, \"message\": \"OK\", \"payload\": { \"age\" : 20, \"status\" : " . "\"online\" }, \"service\": \"Presence\"}"); @@ -91,8 +100,10 @@ public function testApplyStateForChannelsSync() public function testApplyStateForChannelGroup() { - $config = $this->config->clone(); - $config->setUuid("myUUID"); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUserId("myUserId"); $pubnub = new PubNub($config); $setState = new SetStateExposed($pubnub); @@ -100,12 +111,12 @@ public function testApplyStateForChannelGroup() "age" => 20 ]; - $setState->stubFor("/v2/presence/sub-key/demo/channel/,/uuid/myUUID/data") + $setState->stubFor("/v2/presence/sub-key/demo/channel/,/uuid/myUserId/data") ->withQuery([ - "uuid" => "myUUID", "state" => "%7B%22age%22%3A20%7D", + "channel-group" => "cg1", "pnsdk" => $this->encodedSdkName, - "channel-group" => "cg1" + "uuid" => "myUserId", ]) ->setResponseBody("{ \"status\": 200, \"message\": \"OK\", \"payload\": { \"age\" : 20, \"status\" : " . "\"online\" }, \"service\": \"Presence\"}"); @@ -119,8 +130,10 @@ public function testApplyStateForChannelGroup() public function testApplyStateForChannelGroups() { - $config = $this->config->clone(); - $config->setUuid("myUUID"); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUserId("myUserId"); $pubnub = new PubNub($config); $setState = new SetStateExposed($pubnub); @@ -128,9 +141,9 @@ public function testApplyStateForChannelGroups() "age" => 20 ]; - $setState->stubFor("/v2/presence/sub-key/demo/channel/,/uuid/myUUID/data") + $setState->stubFor("/v2/presence/sub-key/demo/channel/,/uuid/myUserId/data") ->withQuery([ - "uuid" => "myUUID", + "uuid" => "myUserId", "state" => "%7B%22age%22%3A20%7D", "pnsdk" => $this->encodedSdkName, "channel-group" => PubNubUtil::urlEncode("cg1,cg2") @@ -147,8 +160,10 @@ public function testApplyStateForChannelGroups() public function testApplyStateForMix() { - $config = $this->config->clone(); - $config->setUuid("myUUID"); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUserId("myUserId"); $pubnub = new PubNub($config); $setState = new SetStateExposed($pubnub); @@ -156,9 +171,9 @@ public function testApplyStateForMix() "age" => 20 ]; - $setState->stubFor("/v2/presence/sub-key/demo/channel/ch1/uuid/myUUID/data") + $setState->stubFor("/v2/presence/sub-key/demo/channel/ch1/uuid/myUserId/data") ->withQuery([ - "uuid" => "myUUID", + "uuid" => "myUserId", "state" => "%7B%22age%22%3A20%7D", "pnsdk" => $this->encodedSdkName, "channel-group" => PubNubUtil::urlEncode("cg1,cg2") @@ -173,45 +188,20 @@ public function testApplyStateForMix() $this->assertEquals($response->getState()["status"], "online"); } - public function testApplyNon200() - { - $this->expectException(PubNubException::class); - - $config = $this->config->clone(); - $config->setUuid("myUUID"); - $pubnub = new PubNub($config); - $setState = new SetStateExposed($pubnub); - - $myState = [ - "age" => 20 - ]; - - $setState->stubFor("/v2/presence/sub-key/demo/channel/ch1/uuid/myUUID/data") - ->withQuery([ - "uuid" => "myUUID", - "state" => "%7B%22age%22%3A20%7D", - "pnsdk" => $this->encodedSdkName, - "channel-group" => PubNubUtil::urlEncode("cg1,cg2") - ]) - ->setResponseBody("{ \"status\": 200, \"message\": \"OK\", \"payload\": { \"age\" : 20, \"status\" : " - . "\"online\" }, \"service\": \"Presence\"}"); - - - $setState->channels("ch1")->channelGroups(["cg1cg2"])->state($myState)->sync(); - } - public function testMissingState() { $this->expectNotToPerformAssertions(); - $config = $this->config->clone(); - $config->setUuid("myUUID"); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUserId("myUserId"); $pubnub = new PubNub($config); $setState = new SetStateExposed($pubnub); - $setState->stubFor("/v2/presence/sub-key/demo/channel/ch1/uuid/myUUID/data") + $setState->stubFor("/v2/presence/sub-key/demo/channel/ch1/uuid/myUserId/data") ->withQuery([ - "uuid" => "myUUID", + "uuid" => "myUserId", "pnsdk" => $this->encodedSdkName, "state" => "%5B%5D" ]) @@ -225,8 +215,10 @@ public function testMissingState() public function testIsAuthRequiredSuccess() { $this->expectNotToPerformAssertions(); - $config = $this->config->clone(); - $config->setUuid("myUUID"); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUuid("myUserId"); $config->setAuthKey("myKey"); $pubnub = new PubNub($config); $setState = new SetStateExposed($pubnub); @@ -235,9 +227,9 @@ public function testIsAuthRequiredSuccess() "age" => 20 ]; - $setState->stubFor("/v2/presence/sub-key/demo/channel/ch1/uuid/myUUID/data") + $setState->stubFor("/v2/presence/sub-key/demo/channel/ch1/uuid/myUserId/data") ->withQuery([ - "uuid" => "myUUID", + "uuid" => "myUserId", "state" => "%7B%22age%22%3A20%7D", "pnsdk" => $this->encodedSdkName, "auth" => "myKey" @@ -253,7 +245,7 @@ public function testNullSubKey() $this->expectException(\TypeError::class); $config = $this->config->clone(); - $config->setUuid("myUUID"); + $config->setUuid("myUserId"); $config->setSubscribeKey(null); $pubnub = new PubNub($config); $setState = new SetStateExposed($pubnub); @@ -262,9 +254,9 @@ public function testNullSubKey() "age" => 20 ]; - $setState->stubFor("/v2/presence/sub-key/demo/channel/ch1/uuid/myUUID/data") + $setState->stubFor("/v2/presence/sub-key/demo/channel/ch1/uuid/myUserId/data") ->withQuery([ - "uuid" => "myUUID", + "uuid" => "myUserId", "state" => "%7B%22age%22%3A20%7D", "pnsdk" => $this->encodedSdkName ]) @@ -276,11 +268,11 @@ public function testNullSubKey() public function testEmptySubKey() { - $this->expectException(PubNubException::class); + $this->expectException(PubNubValidationException::class); $this->expectExceptionMessage("Subscribe Key not configured"); - $config = $this->config->clone(); - $config->setUuid("myUUID"); + $config = new PNConfiguration(); + $config->setUuid("myUserId"); $config->setSubscribeKey(""); $pubnub = new PubNub($config); $setState = new SetStateExposed($pubnub); @@ -289,9 +281,9 @@ public function testEmptySubKey() "age" => 20 ]; - $setState->stubFor("/v2/presence/sub-key/demo/channel/ch1/uuid/myUUID/data") + $setState->stubFor("/v2/presence/sub-key/demo/channel/ch1/uuid/myUserId/data") ->withQuery([ - "uuid" => "myUUID", + "uuid" => "myUserId", "state" => "%7B%22age%22%3A20%7D", "pnsdk" => $this->encodedSdkName ]) @@ -306,8 +298,10 @@ public function testChannelAndGroupMissing() $this->expectException(PubNubException::class); $this->expectExceptionMessage("Channel or group missing"); - $config = $this->config->clone(); - $config->setUuid("myUUID"); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUserId("myUserId"); $pubnub = new PubNub($config); $setState = new SetStateExposed($pubnub); @@ -315,9 +309,9 @@ public function testChannelAndGroupMissing() "age" => 20 ]; - $setState->stubFor("/v2/presence/sub-key/demo/channel/ch1/uuid/myUUID/data") + $setState->stubFor("/v2/presence/sub-key/demo/channel/ch1/uuid/myUserId/data") ->withQuery([ - "uuid" => "myUUID", + "uuid" => "myUserId", "state" => "%7B%22age%22%3A20%7D", "pnsdk" => $this->encodedSdkName ]) @@ -332,18 +326,18 @@ public function testNullPayload() { $this->expectException(PubNubException::class); - $config = $this->config->clone(); - $config->setUuid("myUUID"); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUserId("myUserId"); $pubnub = new PubNub($config); $setState = new SetStateExposed($pubnub); - $myState = [ - "age" => 20 - ]; + $myState = null; - $setState->stubFor("/v2/presence/sub-key/demo/channel/ch1/uuid/myUUID/data") + $setState->stubFor("/v2/presence/sub-key/demo/channel/ch1/uuid/myUserId/data") ->withQuery([ - "uuid" => "myUUID", + "uuid" => "myUserId", "state" => "%7B%22age%22%3A20%7D", "pnsdk" => $this->encodedSdkName ]) @@ -357,24 +351,19 @@ public function testNullPayload() // phpcs:ignore PSR1.Classes.ClassDeclaration class SetStateExposed extends SetState { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function requestOptions() + public function stubFor(string $url): PsrStub { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } diff --git a/tests/integrational/SslTest.php b/tests/integrational/SslTest.php index 42c72b69..c9fce82d 100644 --- a/tests/integrational/SslTest.php +++ b/tests/integrational/SslTest.php @@ -2,6 +2,7 @@ namespace Tests\Integrational; +use PubNub\PNConfiguration; use PubNub\PubNub; class SslTest extends \PubNubTestCase @@ -12,14 +13,13 @@ class SslTest extends \PubNubTestCase */ public function testSslIsSetByDefault() { - $transport = new CheckSslTransport(); - $config = $this->config->clone(); - $config->setTransport($transport); + $config = new PNConfiguration(); + $config->setSubscribeKey('demo'); + $config->setPublishKey('demo'); + $config->setUserId('demo'); $pubnub = new PubNub($config); - $pubnub->time()->sync(); - - $this->assertTrue($transport->isRequestedSecureOrigin()); + $this->assertTrue($pubnub->getConfiguration()->isSecure()); } /** @@ -28,43 +28,13 @@ public function testSslIsSetByDefault() */ public function testSslCanBeDisabled() { - $transport = new CheckSslTransport(); - $config = $this->config->clone(); - $config->setTransport($transport); + $config = new PNConfiguration(); + $config->setSubscribeKey('demo'); + $config->setPublishKey('demo'); + $config->setUserId('demo'); $config->setSecure(false); $pubnub = new PubNub($config); - $pubnub->time()->sync(); - - $this->assertFalse($transport->isRequestedSecureOrigin()); - } -} - -//phpcs:ignore PSR1.Classes.ClassDeclaration -class CheckSslTransport implements \WpOrg\Requests\Transport -{ - protected $requestedThroughHttps; - - public function isRequestedSecureOrigin() - { - return $this->requestedThroughHttps; - } - public function request($url, $headers = array(), $data = array(), $options = array()) - { - $this->requestedThroughHttps = substr($url, 0, 5) === 'https'; - - return "HTTP/1.1 200 OK\r\n" - . "Content-Type: text/javascript; charset=\"UTF-8\"\r\n" - . "Connection: Closed\r\n\r\n" - . "[16614599133417872]"; - } - - // phpcs:ignore PSR1.Methods.CamelCapsMethodName - public function request_multiple($requests, $options) - { - } - - public static function test($capabilities = []) - { + $this->assertFalse($pubnub->getConfiguration()->isSecure()); } } diff --git a/tests/integrational/SubscribePresenceTest.php b/tests/integrational/SubscribePresenceTest.php index 0e8ba106..7b56e966 100644 --- a/tests/integrational/SubscribePresenceTest.php +++ b/tests/integrational/SubscribePresenceTest.php @@ -6,9 +6,11 @@ use PubNub\Enums\PNStatusCategory; use PubNub\Exceptions\PubNubUnsubscribeException; use PubNub\Models\Consumer\PubSub\PNMessageResult; +use PubNub\PNConfiguration; use PubNub\PubNub; use PubNubTestCase; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; /** * Class SubscribePresenceTest @@ -20,49 +22,47 @@ class SubscribePresenceTest extends PubNubTestCase { public function testMessageOnPresenceCallback() { - $transport = new StubTransport(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUuid("myUUID"); + $pubnub = new PubNub($config); + + $client = new PsrStubClient(); + $pubnub->setClient($client); - $transport->stubFor("/v2/presence/sub-key/demo/channel/blah/leave") + $client->addStub((new PsrStub("/v2/presence/sub-key/demo/channel/blah/leave")) ->withQuery([ "pnsdk" => $this->encodedSdkName, "uuid" => "myUUID" ]) - ->setResponseStatus("HTTP/1.0 200 OK") - ->setResponseBody('{"status": 200, "action": "leave", "message": "OK", "service": "Presence"}'); + ->setResponseBody('{"status": 200, "action": "leave", "message": "OK", "service": "Presence"}')); - $transport->stubFor("/v2/subscribe/demo/blah,blah-pnpres/0") + $client->addStub((new PsrStub("/v2/subscribe/demo/blah,blah-pnpres/0")) ->withQuery([ "pnsdk" => $this->encodedSdkName, "uuid" => "myUUID" ]) - ->setResponseStatus("HTTP/1.0 200 OK") - ->setResponseBody('{"t":{"t":"14818963579052943","r":12},"m":[]}'); + ->setResponseBody('{"t":{"t":"14818963579052943","r":12},"m":[]}')); - $transport->stubFor("/v2/subscribe/demo/blah,blah-pnpres/0") + $client->addStub((new PsrStub("/v2/subscribe/demo/blah,blah-pnpres/0")) ->withQuery([ "pnsdk" => $this->encodedSdkName, "uuid" => "myUUID", "tt" => '14818963579052943', "tr" => "12" ]) - ->setResponseStatus("HTTP/1.0 200 OK") ->setResponseBody('{"t":{"t":"14818963588185526","r":12},"m":[{"a":"2","f":0,' . '"p":{"t":"14818963587725382","r":2},"k":"demo","c":"blah-pnpres",' . '"d":{"action": "join", "timestamp": 1481896358, "uuid": "test-subscribe-listener", "occupancy": 1},' - . '"b":"blah-pnpres"}]}'); + . '"b":"blah-pnpres"}]}')); $callback = new MySubscribeCallbackToTestPresence(); - $config = $this->config->clone(); - $config->setTransport($transport); - $config->setUuid("myUUID"); - $pubnub = new PubNub($config); - $pubnub->addListener($callback); $pubnub->subscribe()->channel("blah")->withPresence()->execute(); $this->assertTrue($callback->areBothConnectedAndDisconnectedInvoked()); - $this->assertEquals(3, $transport->requestsCount()); } } diff --git a/tests/integrational/SubscribeSignalTest.php b/tests/integrational/SubscribeSignalTest.php index 3b10c2bb..eab37b63 100644 --- a/tests/integrational/SubscribeSignalTest.php +++ b/tests/integrational/SubscribeSignalTest.php @@ -3,45 +3,46 @@ namespace Tests\Integrational; use PubNub\Callbacks\SubscribeCallback; +use PubNub\PNConfiguration; use PubNub\PubNub; use PubNubTestCase; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class SubscribeSignalTest extends PubNubTestCase { public function testSignal() { - $transport = new StubTransport(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUuid("myUUID"); + $pubnub = new PubNub($config); - $transport->stubFor("/v2/subscribe/demo/test/0") + $client = new PsrStubClient(); + $pubnub->setClient($client); + + $client->addStub((new PsrStub("/v2/subscribe/demo/test/0")) ->withQuery([ "pnsdk" => $this->encodedSdkName, "uuid" => "myUUID" ]) - ->setResponseStatus("HTTP/1.0 200 OK") - ->setResponseBody('{"t":{"t":"14818963579052943","r":12},"m":[]}'); + ->setResponseBody('{"t":{"t":"14818963579052943","r":12},"m":[]}')); - $transport->stubFor("/v2/subscribe/demo/test/0") + $client->addStub((new PsrStub("/v2/subscribe/demo/test/0")) ->withQuery([ "tt" => '14818963579052943', "tr" => "12", "pnsdk" => $this->encodedSdkName, "uuid" => "myUUID" ]) - ->setResponseStatus("HTTP/1.0 200 OK") ->setResponseBody('{"t":{"t":"14921661962885137","r":12},' . '"m":[{"a":"5","f":0,"i":"eda482a8-9de3-4891-b328-b2c1d14f210c",' . '"p":{"t":"14921661962867845","r":12},"k":"demo","e":1,"c":"test","u":{},' - . '"d":{"text":"hey"},"b":"test"}]}'); + . '"d":{"text":"hey"},"b":"test"}]}')); $callback = new MySubscribeCallbackToTestSignal(); - $config = $this->config->clone(); - $config->setTransport($transport); - $config->setUuid("myUUID"); - - $pubnub = new PubNub($config); - $pubnub->addListener($callback); $pubnub->subscribe()->channel("test")->execute(); diff --git a/tests/integrational/SubscribeWildCardTest.php b/tests/integrational/SubscribeWildCardTest.php index 62e18004..6ea7de86 100644 --- a/tests/integrational/SubscribeWildCardTest.php +++ b/tests/integrational/SubscribeWildCardTest.php @@ -9,55 +9,55 @@ use PubNub\PubNub; use PubNubTestCase; use PHPUnit\Framework\AssertionFailedError; -use Tests\Helpers\StubTransport; +use PubNub\PNConfiguration; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class SubscribeWildCardTest extends PubNubTestCase { public function testWildCard() { - $transport = new StubTransport(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUuid("myUUID"); + $pubnub = new PubNub($config); + + $client = new PsrStubClient(); + $pubnub->setClient($client); - $transport->stubFor("/v2/presence/sub-key/demo/channel/channels.%2A/leave") + $client->addStub((new PsrStub("/v2/presence/sub-key/demo/channel/channels.%2A/leave")) ->withQuery([ "pnsdk" => $this->encodedSdkName, "uuid" => "myUUID" ]) - ->setResponseStatus("HTTP/1.0 200 OK") - ->setResponseBody('{"status": 200, "action": "leave", "message": "OK", "service": "Presence"}'); + ->setResponseBody('{"status": 200, "action": "leave", "message": "OK", "service": "Presence"}')); - $transport->stubFor("/v2/subscribe/demo/channels.%2A/0") + $client->addStub((new PsrStub("/v2/subscribe/demo/channels.%2A/0")) ->withQuery([ "pnsdk" => $this->encodedSdkName, "uuid" => "myUUID" ]) - ->setResponseStatus("HTTP/1.0 200 OK") - ->setResponseBody('{"t":{"t":"14818963579052943","r":12},"m":[]}'); + ->setResponseBody('{"t":{"t":"14818963579052943","r":12},"m":[]}')); - $transport->stubFor("/v2/subscribe/demo/channels.%2A/0") + $client->addStub((new PsrStub("/v2/subscribe/demo/channels.%2A/0")) ->withQuery([ "tt" => '14818963579052943', "tr" => "12", "pnsdk" => $this->encodedSdkName, "uuid" => "myUUID" ]) - ->setResponseStatus("HTTP/1.0 200 OK") ->setResponseBody('{"t":{"t":"14921661962885137","r":12},' . '"m":[{"a":"5","f":0,"i":"eda482a8-9de3-4891-b328-b2c1d14f210c",' . '"p":{"t":"14921661962867845","r":12},"k":"demo","c":"channels.one","u":{},' - . '"d":{"text":"hey"},"b":"channels.*"}]}'); + . '"d":{"text":"hey"},"b":"channels.*"}]}')); $callback = new MySubscribeCallbackToTestWildCard(); - $config = $this->config->clone(); - $config->setTransport($transport); - $config->setUuid("myUUID"); - $pubnub = new PubNub($config); - $pubnub->addListener($callback); $pubnub->subscribe()->channel("channels.*")->execute(); $this->assertTrue($callback->areBothConnectedAndDisconnectedInvoked()); - $this->assertEquals(3, $transport->requestsCount()); } } diff --git a/tests/integrational/WhereNowTest.php b/tests/integrational/WhereNowTest.php index cbfd904e..ce50f1a6 100644 --- a/tests/integrational/WhereNowTest.php +++ b/tests/integrational/WhereNowTest.php @@ -5,8 +5,11 @@ use PubNub\Endpoints\Presence\WhereNow; use PubNub\Exceptions\PubNubResponseParsingException; use PubNubTestCase; -use Tests\Helpers\StubTransport; use PHPUnit\Framework\Constraint\IsType; +use PubNub\PNConfiguration; +use PubNub\PubNub; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class WhereNowTest extends PubNubTestCase { @@ -14,16 +17,22 @@ public function testSuccess() { $uuid = "where-now-uuid"; - $this->pubnub_demo->getConfiguration()->setUuid($uuid); - - $whereNow = new WhereNowTestExposed($this->pubnub_demo); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUuid($uuid); + $pubnub = new PubNub($config); + $whereNow = new WhereNowTestExposed($pubnub); $whereNow->stubFor("/v2/presence/sub-key/demo/uuid/where-now-uuid") ->withQuery([ 'uuid' => $uuid, 'pnsdk' => $this->encodedSdkName ]) - ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {\"channels\": [\"a\",\"b\"]}, \"service\": \"Presence\"}"); + ->setResponseBody( + "{\"status\": 200, \"message\": \"OK\", \"payload\": {\"channels\": [\"a\",\"b\"]}, " + . "\"service\": \"Presence\"}" + ); $response = $whereNow->sync(); @@ -35,16 +44,22 @@ public function testSuccessCustomUUID() $uuid = "where-now-uuid"; $customUuid = "custom-uuid"; - $this->pubnub_demo->getConfiguration()->setUuid($uuid); - - $whereNow = new WhereNowTestExposed($this->pubnub_demo); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUuid($uuid); + $pubnub = new PubNub($config); + $whereNow = new WhereNowTestExposed($pubnub); $whereNow->stubFor("/v2/presence/sub-key/demo/uuid/custom-uuid") ->withQuery([ 'uuid' => $uuid, 'pnsdk' => $this->encodedSdkName ]) - ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {\"channels\": [\"a\",\"b\"]}, \"service\": \"Presence\"}"); + ->setResponseBody( + "{\"status\": 200, \"message\": \"OK\", \"payload\": {\"channels\": [\"a\",\"b\"]}, " + . "\"service\": \"Presence\"}" + ); $response = $whereNow->uuid($customUuid)->sync(); @@ -56,15 +71,22 @@ public function testBrokenWithString() $this->expectException(PubNubResponseParsingException::class); $this->expectExceptionMessage("Unable to parse server response: Syntax error"); - $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); - $whereNow = new WhereNowTestExposed($this->pubnub_demo); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUuid("myUUID"); + $pubnub = new PubNub($config); + $whereNow = new WhereNowTestExposed($pubnub); $whereNow->stubFor("/v2/presence/sub-key/demo/uuid/myUUID") ->withQuery([ 'uuid' => "myUUID", 'pnsdk' => $this->encodedSdkName ]) - ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {\"channels\": [zimp]}, \"service\": \"Presence\"}"); + ->setResponseBody( + "{\"status\": 200, \"message\": \"OK\", \"payload\": {\"channels\": [zimp]}, \"service\": " + . "\"Presence\"}" + ); $whereNow->sync(); } @@ -74,15 +96,22 @@ public function testBrokenWithoutJSON() $this->expectException(PubNubResponseParsingException::class); $this->expectExceptionMessage("Unable to parse server response: Syntax error"); - $this->pubnub_demo->getConfiguration()->setUuid("myUUID"); - $whereNow = new WhereNowTestExposed($this->pubnub_demo); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUuid("myUUID"); + $pubnub = new PubNub($config); + $whereNow = new WhereNowTestExposed($pubnub); $whereNow->stubFor("/v2/presence/sub-key/demo/uuid/myUUID") ->withQuery([ 'uuid' => "myUUID", 'pnsdk' => $this->encodedSdkName ]) - ->setResponseBody("{\"status\": 200, \"message\": \"OK\", \"payload\": {\"channels\": zimp}, \"service\": \"Presence\"}"); + ->setResponseBody( + "{\"status\": 200, \"message\": \"OK\", \"payload\": {\"channels\": zimp}, \"service\": " + . "\"Presence\"}" + ); $whereNow->sync(); } @@ -114,27 +143,22 @@ public function testSuperCall() } } - +// phpcs:ignore PSR1.Classes.ClassDeclaration class WhereNowTestExposed extends WhereNow { - protected $transport; + protected PsrStubClient $client; - public function __construct($pubnubInstance) + public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function stubFor($url) + public function stubFor(string $url): PsrStub { - return $this->transport->stubFor($url); + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } - - public function requestOptions() - { - return [ - 'transport' => $this->transport - ]; - } -} \ No newline at end of file +} diff --git a/tests/integrational/objects/member/MembersHappyPathTest.php b/tests/integrational/objects/member/MembersHappyPathTest.php index 1faf3e5b..a1d0587b 100644 --- a/tests/integrational/objects/member/MembersHappyPathTest.php +++ b/tests/integrational/objects/member/MembersHappyPathTest.php @@ -68,7 +68,7 @@ public function testHappyPath(): void $this->assertInstanceOf(PNSetUUIDMetadataResult::class, $userSetup2); $includes = new PNMemberIncludes(); - $includes->user()->userId()->userCustom()->userType()->userStatus()->custom()->status()->type(); + $includes->user()->user()->userCustom()->userType()->userStatus()->custom()->status()->type(); $addMembers = $this->pubnub->setMembers() ->channel($this->channel) diff --git a/tests/integrational/objects/membership/MembershipsHappyPathTest.php b/tests/integrational/objects/membership/MembershipsHappyPathTest.php index 65fdff3a..607557a9 100644 --- a/tests/integrational/objects/membership/MembershipsHappyPathTest.php +++ b/tests/integrational/objects/membership/MembershipsHappyPathTest.php @@ -31,7 +31,7 @@ public function testHappyPath(): void sleep(1); $includes = new PNMembershipIncludes(); - $includes->channel()->channelId()->channelCustom()->channelType()->channelStatus()->custom()->status()->type(); + $includes->channel()->channel()->channelCustom()->channelType()->channelStatus()->custom()->status()->type(); $addMembership = $this->pubnub->setMemberships() ->userId($this->user) ->memberships([ diff --git a/tests/integrational/push/AddChannelsToPushEndpointTest.php b/tests/integrational/push/AddChannelsToPushEndpointTest.php index 4c0a24bb..cb5b149c 100644 --- a/tests/integrational/push/AddChannelsToPushEndpointTest.php +++ b/tests/integrational/push/AddChannelsToPushEndpointTest.php @@ -4,17 +4,22 @@ use PubNub\Endpoints\Push\AddChannelsToPush; use PubNub\Enums\PNPushType; +use PubNub\PNConfiguration; use PubNub\PubNub; use PubNubTestCase; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class AddChannelsToPushEndpointTest extends PubNubTestCase { public function testPushAddSingleChannel() { - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); - - $add = new AddChannelsToPushEndpointExposed($this->pubnub); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUuid("sampleUUID"); + $pubnub = new PubNub($config); + $add = new AddChannelsToPushEndpointExposed($pubnub); $add->stubFor("/v1/push/sub-key/demo/devices/coolDevice") ->withQuery([ @@ -35,9 +40,12 @@ public function testPushAddSingleChannel() public function testPushAddMultipleChannels() { - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); - - $add = new AddChannelsToPushEndpointExposed($this->pubnub); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUuid("sampleUUID"); + $pubnub = new PubNub($config); + $add = new AddChannelsToPushEndpointExposed($pubnub); $add->stubFor("/v1/push/sub-key/demo/devices/coolDevice") ->withQuery([ @@ -58,9 +66,12 @@ public function testPushAddMultipleChannels() public function testPushAddApns2() { - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); - - $add = new AddChannelsToPushEndpointExposed($this->pubnub); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUuid("sampleUUID"); + $pubnub = new PubNub($config); + $add = new AddChannelsToPushEndpointExposed($pubnub); $add->stubFor("/v2/push/sub-key/demo/devices-apns2/coolDevice") ->withQuery([ @@ -84,9 +95,12 @@ public function testPushAddApns2() public function testPushAddFCM() { - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); - - $add = new AddChannelsToPushEndpointExposed($this->pubnub); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUuid("sampleUUID"); + $pubnub = new PubNub($config); + $add = new AddChannelsToPushEndpointExposed($pubnub); $add->stubFor("/v1/push/sub-key/demo/devices/coolDevice") ->withQuery([ @@ -114,9 +128,12 @@ public function testWarningWhenUsingDeprecatedGCMType() $this->expectException(\Exception::class); $this->expectExceptionMessage('GCM is deprecated. Please use FCM instead.'); - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); - - $add = new AddChannelsToPushEndpointExposed($this->pubnub); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); + $config->setUuid("sampleUUID"); + $pubnub = new PubNub($config); + $add = new AddChannelsToPushEndpointExposed($pubnub); $add->stubFor("/v1/push/sub-key/demo/devices/coolDevice") ->withQuery([ @@ -139,24 +156,19 @@ public function testWarningWhenUsingDeprecatedGCMType() // phpcs:ignore PSR1.Classes.ClassDeclaration class AddChannelsToPushEndpointExposed extends AddChannelsToPush { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function requestOptions() + public function stubFor(string $url): PsrStub { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } diff --git a/tests/integrational/push/ListPushProvisionsEndpointTest.php b/tests/integrational/push/ListPushProvisionsEndpointTest.php index d22bc811..3ed1d19b 100644 --- a/tests/integrational/push/ListPushProvisionsEndpointTest.php +++ b/tests/integrational/push/ListPushProvisionsEndpointTest.php @@ -4,15 +4,19 @@ use PubNub\Endpoints\Push\ListPushProvisions; use PubNub\Enums\PNPushType; +use PubNub\PNConfiguration; use PubNub\PubNub; use PubNubTestCase; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class ListPushProvisionsEndpointTest extends PubNubTestCase { public function testListChannelGroupAPNS() { - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); $list = new ListPushProvisionsEndpointExposed($pubnub); @@ -34,7 +38,9 @@ public function testListChannelGroupAPNS() public function testListChannelGroupAPNS2() { - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); $list = new ListPushProvisionsEndpointExposed($pubnub); @@ -59,9 +65,9 @@ public function testListChannelGroupAPNS2() public function testListChannelGroupFCM() { - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); - - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); $list = new ListPushProvisionsEndpointExposed($pubnub); @@ -83,7 +89,9 @@ public function testListChannelGroupFCM() public function testListChannelGroupMPNS() { - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); $list = new ListPushProvisionsEndpointExposed($pubnub); @@ -112,7 +120,9 @@ public function testWarningWhenUsingDeprecatedGCMType() $this->expectException(\Exception::class); $this->expectExceptionMessage('GCM is deprecated. Please use FCM instead.'); - $config = $this->config->clone(); + $config = new PNConfiguration(); + $config->setSubscribeKey("demo"); + $config->setPublishKey("demo"); $config->setUuid("sampleUUID"); $pubnub = new PubNub($config); $list = new ListPushProvisionsEndpointExposed($pubnub); @@ -136,34 +146,19 @@ public function testWarningWhenUsingDeprecatedGCMType() // phpcs:ignore PSR1.Classes.ClassDeclaration class ListPushProvisionsEndpointExposed extends ListPushProvisions { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); - } - - public function buildParams() - { - return parent::buildParams(); - } - - public function buildPath() - { - return parent::buildPath(); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function requestOptions() + public function stubFor(string $url): PsrStub { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } diff --git a/tests/integrational/push/RemoveChannelsFromPushEndpointTest.php b/tests/integrational/push/RemoveChannelsFromPushEndpointTest.php index 2bd090e8..ee90315a 100644 --- a/tests/integrational/push/RemoveChannelsFromPushEndpointTest.php +++ b/tests/integrational/push/RemoveChannelsFromPushEndpointTest.php @@ -5,22 +5,23 @@ use PubNub\Endpoints\Push\RemoveChannelsFromPush; use PubNub\Enums\PNPushType; use PubNub\PubNub; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class RemoveChannelsFromPushEndpointTest extends \PubNubTestCase { public function testPushRemoveSingleChannel() { - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("sampleUUID"); - $remove = new RemoveChannelsFromPushEndpointExposed($this->pubnub); + $remove = new RemoveChannelsFromPushEndpointExposed($this->pubnub_demo); $remove->stubFor("/v1/push/sub-key/demo/devices/coolDevice") ->withQuery([ - "pnsdk" => $this->encodedSdkName, + "remove" => "ch", "type" => "apns", + "pnsdk" => $this->pubnub_demo->getSdkFullName(), "uuid" => "sampleUUID", - "remove" => "ch" ]) ->setResponseBody('[1, "Modified Channels"]'); @@ -33,17 +34,17 @@ public function testPushRemoveSingleChannel() } public function testPushRemoveApns2() { - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("sampleUUID"); - $remove = new RemoveChannelsFromPushEndpointExposed($this->pubnub); + $remove = new RemoveChannelsFromPushEndpointExposed($this->pubnub_demo); $remove->stubFor("/v2/push/sub-key/demo/devices-apns2/coolDevice") ->withQuery([ - "pnsdk" => $this->encodedSdkName, - "uuid" => "sampleUUID", "remove" => "ch", "topic" => "coolTopic", - "environment" => "development" + "environment" => "development", + "pnsdk" => $this->pubnub_demo->getSdkFullName(), + "uuid" => "sampleUUID", ]) ->setResponseBody('[1, "Modified Channels"]'); @@ -58,16 +59,16 @@ public function testPushRemoveApns2() public function testPushRemoveMultipleChannels() { - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("sampleUUID"); - $remove = new RemoveChannelsFromPushEndpointExposed($this->pubnub); + $remove = new RemoveChannelsFromPushEndpointExposed($this->pubnub_demo); $remove->stubFor("/v1/push/sub-key/demo/devices/coolDevice") ->withQuery([ - "pnsdk" => $this->encodedSdkName, + "remove" => "ch1,ch2", "type" => "mpns", + "pnsdk" => $this->pubnub_demo->getSdkFullName(), "uuid" => "sampleUUID", - "remove" => "ch1,ch2" ]) ->setResponseBody('[1, "Modified Channels"]'); @@ -81,16 +82,16 @@ public function testPushRemoveMultipleChannels() public function testPushRemoveFCM() { - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("sampleUUID"); - $remove = new RemoveChannelsFromPushEndpointExposed($this->pubnub); + $remove = new RemoveChannelsFromPushEndpointExposed($this->pubnub_demo); $remove->stubFor("/v1/push/sub-key/demo/devices/coolDevice") ->withQuery([ - "pnsdk" => $this->encodedSdkName, + "remove" => "ch1,ch2,ch3", "type" => "gcm", + "pnsdk" => $this->pubnub_demo->getSdkFullName(), "uuid" => "sampleUUID", - "remove" => "ch1,ch2,ch3" ]) ->setResponseBody('[1, "Modified Channels"]'); @@ -106,24 +107,19 @@ public function testPushRemoveFCM() // phpcs:ignore PSR1.Classes.ClassDeclaration class RemoveChannelsFromPushEndpointExposed extends RemoveChannelsFromPush { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function requestOptions() + public function stubFor(string $url): PsrStub { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } diff --git a/tests/integrational/push/RemoveDeviceFromPushEndpointTest.php b/tests/integrational/push/RemoveDeviceFromPushEndpointTest.php index 5d26aa34..1cd0c426 100644 --- a/tests/integrational/push/RemoveDeviceFromPushEndpointTest.php +++ b/tests/integrational/push/RemoveDeviceFromPushEndpointTest.php @@ -5,20 +5,21 @@ use PubNub\Endpoints\Push\RemoveDeviceFromPush; use PubNub\Enums\PNPushType; use PubNub\PubNub; -use Tests\Helpers\StubTransport; +use PubNubTests\helpers\PsrStub; +use PubNubTests\helpers\PsrStubClient; class RemoveDeviceFromPushEndpointTest extends \PubNubTestCase { public function testRemovePushAPNS() { - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("sampleUUID"); - $remove = new RemoveDeviceFromPushEndpointExposed($this->pubnub); + $remove = new RemoveDeviceFromPushEndpointExposed($this->pubnub_demo); $remove->stubFor("/v1/push/sub-key/demo/devices/coolDevice/remove") ->withQuery([ - "pnsdk" => $this->encodedSdkName, "type" => "apns", + "pnsdk" => $this->pubnub_demo->getSdkFullName(), "uuid" => "sampleUUID", ]) ->setResponseBody('[1, "Modified Channels"]'); @@ -32,16 +33,16 @@ public function testRemovePushAPNS() public function testRemovePushAPNS2() { - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("sampleUUID"); - $remove = new RemoveDeviceFromPushEndpointExposed($this->pubnub); + $remove = new RemoveDeviceFromPushEndpointExposed($this->pubnub_demo); $remove->stubFor("/v2/push/sub-key/demo/devices-apns2/coolDevice/remove") ->withQuery([ - "pnsdk" => $this->encodedSdkName, - "uuid" => "sampleUUID", "topic" => "coolTopic", - "environment" => "production" + "environment" => "production", + "pnsdk" => $this->pubnub_demo->getSdkFullName(), + "uuid" => "sampleUUID", ]) ->setResponseBody('[1, "Modified Channels"]'); @@ -56,14 +57,14 @@ public function testRemovePushAPNS2() public function testRemovePushMPNS() { - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("sampleUUID"); - $remove = new RemoveDeviceFromPushEndpointExposed($this->pubnub); + $remove = new RemoveDeviceFromPushEndpointExposed($this->pubnub_demo); $remove->stubFor("/v1/push/sub-key/demo/devices/coolDevice/remove") ->withQuery([ - "pnsdk" => $this->encodedSdkName, "type" => "mpns", + "pnsdk" => $this->pubnub_demo->getSdkFullName(), "uuid" => "sampleUUID", ]) ->setResponseBody('[1, "Modified Channels"]'); @@ -77,14 +78,14 @@ public function testRemovePushMPNS() public function testRemovePushFCM() { - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("sampleUUID"); - $remove = new RemoveDeviceFromPushEndpointExposed($this->pubnub); + $remove = new RemoveDeviceFromPushEndpointExposed($this->pubnub_demo); $remove->stubFor("/v1/push/sub-key/demo/devices/coolDevice/remove") ->withQuery([ - "pnsdk" => $this->encodedSdkName, "type" => "gcm", + "pnsdk" => $this->pubnub_demo->getSdkFullName(), "uuid" => "sampleUUID", ]) ->setResponseBody('[1, "Modified Channels"]'); @@ -104,14 +105,14 @@ public function testWarningWhenUsingDeprecatedGCMType() $this->expectException(\Exception::class); $this->expectExceptionMessage('GCM is deprecated. Please use FCM instead.'); - $this->pubnub->getConfiguration()->setUuid("sampleUUID"); + $this->pubnub_demo->getConfiguration()->setUuid("sampleUUID"); - $remove = new RemoveDeviceFromPushEndpointExposed($this->pubnub); + $remove = new RemoveDeviceFromPushEndpointExposed($this->pubnub_demo); $remove->stubFor("/v1/push/sub-key/demo/devices/coolDevice/remove") ->withQuery([ - "pnsdk" => $this->encodedSdkName, "type" => "gcm", + "pnsdk" => $this->pubnub_demo->getSdkFullName(), "uuid" => "sampleUUID", ]) ->setResponseBody('[1, "Modified Channels"]'); @@ -127,24 +128,19 @@ public function testWarningWhenUsingDeprecatedGCMType() // phpcs:ignore PSR1.Classes.ClassDeclaration class RemoveDeviceFromPushEndpointExposed extends RemoveDeviceFromPush { - protected $transport; + protected PsrStubClient $client; public function __construct(PubNub $pubnubInstance) { parent::__construct($pubnubInstance); - - $this->transport = new StubTransport(); - } - - public function stubFor($url) - { - return $this->transport->stubFor($url); + $this->client = new PsrStubClient(); + $pubnubInstance->setClient($this->client); } - public function requestOptions() + public function stubFor(string $url): PsrStub { - return [ - 'transport' => $this->transport - ]; + $stub = new PsrStub($url); + $this->client->addStub($stub); + return $stub; } } diff --git a/tests/unit/StubTest.php b/tests/unit/StubTest.php deleted file mode 100644 index 189e7c3d..00000000 --- a/tests/unit/StubTest.php +++ /dev/null @@ -1,60 +0,0 @@ -withQuery([ - 'uuid' => "blah", - 'pnsdk' => '123' - ]); - $this->assertTrue($stub->isQueryMatch("uuid=blah&pnsdk=123")); - } - - /** - * @group stub - */ - public function testAny() - { - $stub = (new Stub(""))->withQuery([ - 'uuid' => Stub::ANY, - 'pnsdk' => '123' - ]); - - $this->assertTrue($stub->isQueryMatch("uuid=blahblah123&pnsdk=123")); - } - - /** - * @group stub - */ - public function testExtraExpectedArgument() - { - $stub = (new Stub(""))->withQuery([ - 'id' => '15', - 'uuid' => 'blah', - 'pnsdk' => '123' - ]); - - $this->assertFalse($stub->isQueryMatch("uuid=blah&pnsdk=123")); - } - - /** - * @group stub - */ - public function testExtraActualArgument() - { - $stub = (new Stub(""))->withQuery([ - 'uuid' => 'blah', - 'pnsdk' => '123' - ]); - - $this->assertFalse($stub->isQueryMatch("uuid=blah&pnsdk=123&id=321")); - } -}