diff --git a/.gitattributes b/.gitattributes
index f7c0249..e4189e2 100644
--- a/.gitattributes
+++ b/.gitattributes
@@ -2,15 +2,16 @@
# Exclude these files from release archives.
#
# This will also make the files unavailable when using Composer with `--prefer-dist`.
-# If you develop using Composer, use `--prefer-source`.
#
# Via WPCS.
#
-/phpcs.xml export-ignore
-/phpunit.xml export-ignore
/.github export-ignore
-/.php-cs-fixer.dist.php export-ignore
+/.phpcs.xml export-ignore
+/.phpcs export-ignore
+/phpunit.xml export-ignore
/tests export-ignore
+/configure.php export-ignore
+/Makefile export-ignore
#
# Auto detect text files and perform LF normalization.
diff --git a/.github/workflows/all-pr-tests.yml b/.github/workflows/all-pr-tests.yml
new file mode 100644
index 0000000..3a10dc2
--- /dev/null
+++ b/.github/workflows/all-pr-tests.yml
@@ -0,0 +1,45 @@
+name: "All Pull Request Tests"
+
+on:
+ pull_request:
+ branches:
+ - main
+ types: [opened, synchronize, reopened, ready_for_review]
+
+jobs:
+ # We use a single job to ensure that all steps run in the same environment and
+ # reduce the number of minutes used.
+ pr-tests:
+ # Don't run on draft PRs
+ if: github.event.pull_request.draft == false
+ # Timeout after 10 minutes
+ timeout-minutes: 10
+ # Define a matrix of PHP/WordPress versions to test against
+ strategy:
+ matrix:
+ php: [8.2, 8.3]
+ wordpress: ["latest"]
+ multisite: [false, true]
+ runs-on: ubuntu-latest
+ # Cancel any existing runs of this workflow
+ concurrency:
+ group: ${{ github.workflow }}-${{ github.event_name }}-${{ github.event.pull_request.number || github.ref }}-P${{ matrix.php }}-WP${{ matrix.wordpress }}-MS${{ matrix.multisite }}
+ cancel-in-progress: true
+ # Name the job in the matrix
+ name: "PR Tests PHP ${{ matrix.php }} WordPress ${{ matrix.wordpress }} multisite ${{ matrix.multisite }}"
+ steps:
+ - uses: actions/checkout@v4
+
+ - name: Run General Tests
+ # See https://github.com/alleyinteractive/action-test-general for more options
+ uses: alleyinteractive/action-test-general@develop
+
+ - name: Run PHP Tests
+ # See https://github.com/alleyinteractive/action-test-php for more options
+ uses: alleyinteractive/action-test-php@develop
+ with:
+ php-version: '${{ matrix.php }}'
+ audit-command: 'composer audit --no-dev --ansi --no-interaction'
+ wordpress-version: '${{ matrix.wordpress }}'
+ wordpress-multisite: '${{ matrix.multisite }}'
+ skip-wordpress-install: 'true'
diff --git a/.github/workflows/coding-standards.yml b/.github/workflows/coding-standards.yml
deleted file mode 100644
index bd66de4..0000000
--- a/.github/workflows/coding-standards.yml
+++ /dev/null
@@ -1,12 +0,0 @@
-name: Coding Standards
-
-on:
- pull_request:
- branches:
- - main
- types: [opened, synchronize, reopened, ready_for_review]
-
-jobs:
- coding-standards:
- if: github.event.pull_request.draft == false
- uses: alleyinteractive/.github/.github/workflows/php-coding-standards.yml@main
diff --git a/.github/workflows/unit-test.yml b/.github/workflows/unit-test.yml
deleted file mode 100644
index de6d38b..0000000
--- a/.github/workflows/unit-test.yml
+++ /dev/null
@@ -1,21 +0,0 @@
-name: Testing Suite
-
-on:
- pull_request:
- branches:
- - main
- types: [opened, synchronize, reopened, ready_for_review]
-
-jobs:
- php-tests:
- if: github.event.pull_request.draft == false
- strategy:
- matrix:
- php: ["8.2", "8.1", "8.0"]
- wordpress: ["latest"]
- multisite: [false, true]
- uses: alleyinteractive/.github/.github/workflows/php-tests.yml@main
- with:
- php: ${{ matrix.php }}
- wordpress: ${{ matrix.wordpress }}
- multisite: ${{ matrix.multisite }}
diff --git a/.php-cs-fixer.dist.php b/.php-cs-fixer.dist.php
deleted file mode 100644
index ff95cb5..0000000
--- a/.php-cs-fixer.dist.php
+++ /dev/null
@@ -1,36 +0,0 @@
-
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- *
- * @package wp-find-one
- */
-
-$finder = PhpCsFixer\Finder::create()->in(
- [
- __DIR__ . '/src/',
- __DIR__ . '/tests/',
- ]
-);
-
-$config = new PhpCsFixer\Config();
-$config->setRules(
- [
- '@PHP81Migration' => true,
- // Enabled by '@PHP81Migration' but generates invalid spacing for WordPress.
- 'method_argument_space' => false,
-
- 'final_class' => true,
- 'native_constant_invocation' => true,
- 'native_function_casing' => true,
- 'native_function_invocation' => true,
- 'native_function_type_declaration_casing' => true,
- ]
-);
-$config->setFinder( $finder );
-
-return $config;
diff --git a/phpcs.xml b/.phpcs.xml
similarity index 62%
rename from phpcs.xml
rename to .phpcs.xml
index f5a5dc8..bee6944 100644
--- a/phpcs.xml
+++ b/.phpcs.xml
@@ -2,9 +2,8 @@
PHP_CodeSniffer standard for alleyinteractive/wp-find-one.
- src/
- tests/
- vendor/
+
+
-
-
+ src/
+ tests/
+ vendor/
+
+
+
-
-
+
+
-
-
-
-
-
-
+
+
-
+
-
+
+
+
-
-
- tests/
+
+ tests/*
-
-
-
diff --git a/CHANGELOG.md b/CHANGELOG.md
index f9f211e..1eeb13e 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -6,6 +6,12 @@ This library adheres to [Semantic Versioning](https://semver.org/) and [Keep a C
Nothing yet.
+## 3.0.0
+
+### Changed
+
+- The minimum PHP version is now 8.2.
+
## 2.0.0
### Removed
diff --git a/README.md b/README.md
index 93857c8..fbad800 100644
--- a/README.md
+++ b/README.md
@@ -7,7 +7,7 @@ The `find_one()` family of functions are wrappers for common WordPress retrieval
Install the latest version with:
```bash
-$ composer require alleyinteractive/wp-find-one
+composer require alleyinteractive/wp-find-one
```
## Basic usage
diff --git a/composer.json b/composer.json
index e175f0e..edce510 100644
--- a/composer.json
+++ b/composer.json
@@ -1,49 +1,68 @@
{
- "name": "alleyinteractive/wp-find-one",
- "description": "Query for and return one WordPress post, term, or other object, bypassing intermediate arrays.",
- "type": "library",
- "license": "GPL-2.0-or-later",
- "authors": [
- {
- "name": "Alley",
- "email": "info@alley.com"
- }
- ],
- "autoload": {
- "files": [
- "src/alley/wp/find-one.php",
- "src/alley/wp/find-result.php"
- ]
- },
- "config": {
- "allow-plugins": {
- "alleyinteractive/composer-wordpress-autoloader": true,
- "dealerdirect/phpcodesniffer-composer-installer": true
+ "name": "alleyinteractive/wp-find-one",
+ "description": "Query for and return one WordPress post, term, or other object, bypassing intermediate arrays.",
+ "license": "GPL-2.0-or-later",
+ "type": "library",
+ "authors": [
+ {
+ "name": "Alley",
+ "email": "info@alley.com"
+ }
+ ],
+ "require": {
+ "php": "^8.2"
+ },
+ "require-dev": {
+ "alleyinteractive/alley-coding-standards": "^2.0",
+ "ergebnis/composer-normalize": "^2.44",
+ "mantle-framework/testkit": "^1.2",
+ "szepeviktor/phpstan-wordpress": "^1.3"
+ },
+ "minimum-stability": "dev",
+ "prefer-stable": true,
+ "autoload": {
+ "files": [
+ "src/find-one.php",
+ "src/find-result.php"
+ ]
+ },
+ "autoload-dev": {
+ "psr-4": {
+ "Alley\\WP\\Tests\\": "tests/"
+ }
+ },
+ "config": {
+ "allow-plugins": {
+ "alleyinteractive/composer-wordpress-autoloader": true,
+ "dealerdirect/phpcodesniffer-composer-installer": true,
+ "ergebnis/composer-normalize": true
+ },
+ "lock": false,
+ "sort-packages": true
+ },
+ "extra": {
+ "wordpress-autoloader": {
+ "autoload": {
+ "Alley\\WP\\": "src/"
+ }
+ }
},
- "lock": false
- },
- "require": {
- "php": "^8.0"
- },
- "require-dev": {
- "alleyinteractive/alley-coding-standards": "^1.0.0",
- "friendsofphp/php-cs-fixer": "^3.8",
- "mantle-framework/testkit": "^0.9"
- },
- "scripts": {
- "fixer": "php-cs-fixer -v fix --allow-risky=yes",
- "phpcbf": "phpcbf",
- "phpcs": "phpcs",
- "phpunit": "phpunit"
- },
- "extra": {
- "wordpress-autoloader": {
- "autoload": {
- "Alley\\": "src/alley/"
- },
- "autoload-dev": {
- "Alley\\": "tests/alley/"
- }
+ "scripts": {
+ "pre-install-cmd": [
+ "@tidy"
+ ],
+ "post-install-cmd": [
+ "@tidy"
+ ],
+ "phpcbf": "phpcbf",
+ "phpcs": "phpcs",
+ "phpstan": "phpstan -v --memory-limit=512M",
+ "phpunit": "phpunit",
+ "test": [
+ "@phpcs",
+ "@phpstan",
+ "@phpunit"
+ ],
+ "tidy": "[ $COMPOSER_DEV_MODE -eq 0 ] || composer normalize"
}
- }
}
diff --git a/phpstan.neon b/phpstan.neon
new file mode 100644
index 0000000..f8d6303
--- /dev/null
+++ b/phpstan.neon
@@ -0,0 +1,9 @@
+includes:
+ - vendor/szepeviktor/phpstan-wordpress/extension.neon
+
+parameters:
+ # Level 9 is the highest level
+ level: max
+
+ paths:
+ - src/
diff --git a/phpunit.xml b/phpunit.xml
index 90707de..7b6cd83 100644
--- a/phpunit.xml
+++ b/phpunit.xml
@@ -1,12 +1,20 @@
+
-
- tests/alley/wp/
-
+ xsi:noNamespaceSchemaLocation="https://schema.phpunit.de/11.2/phpunit.xsd"
+ cacheDirectory=".phpunit.result.cache">
+
+
+ tests/Feature
+
+
+ tests/Unit
+
+
+
+
+
+
diff --git a/src/alley/wp/find-one.php b/src/find-one.php
similarity index 72%
rename from src/alley/wp/find-one.php
rename to src/find-one.php
index 8224c33..0e6cc53 100644
--- a/src/alley/wp/find-one.php
+++ b/src/find-one.php
@@ -15,7 +15,7 @@
/**
* Query for and return one post.
*
- * @param array $args Query arguments.
+ * @param mixed[] $args Query arguments.
* @return mixed|null A \WP_Post object, an alternate return type if requested
* in $args, or null.
*/
@@ -23,7 +23,8 @@ function find_one_post( array $args ) {
return find_result(
\WP_Post::class,
get_posts( // phpcs:ignore WordPressVIPMinimum.Functions.RestrictedFunctions.get_posts_get_posts
- array_merge(
+ // Not attempting to re-enumerate all documented arguments.
+ array_merge( // @phpstan-ignore argument.type
$args,
[
'posts_per_page' => 1,
@@ -37,7 +38,7 @@ function find_one_post( array $args ) {
/**
* Query for and return one term.
*
- * @param array $args Query arguments.
+ * @param mixed[] $args Query arguments.
* @return mixed|null A \WP_Term object, an alternate return type if requested
* in $args, or null.
*/
@@ -45,7 +46,8 @@ function find_one_term( array $args ) {
return find_result(
\WP_Term::class,
get_terms(
- array_merge(
+ // Not attempting to re-enumerate all documented arguments.
+ array_merge( // @phpstan-ignore argument.type
$args,
[
'number' => 1,
@@ -58,7 +60,7 @@ function find_one_term( array $args ) {
/**
* Query for and return one comment.
*
- * @param array $args Query arguments.
+ * @param mixed[] $args Query arguments.
* @return mixed|null A \WP_Comment object, an alternate return type if
* requested in $args, or null.
*/
@@ -66,7 +68,8 @@ function find_one_comment( array $args ) {
return find_result(
\WP_Comment::class,
get_comments(
- array_merge(
+ // Not attempting to re-enumerate all documented arguments.
+ array_merge( // @phpstan-ignore argument.type
$args,
[
'number' => 1,
@@ -79,7 +82,7 @@ function find_one_comment( array $args ) {
/**
* Query for and return one user.
*
- * @param array $args Query arguments.
+ * @param mixed[] $args Query arguments.
* @return mixed|null A \WP_User object, an alternate return type if requested
* in $args, or null.
*/
@@ -87,7 +90,8 @@ function find_one_user( array $args ) {
return find_result(
\WP_User::class,
get_users(
- array_merge(
+ // Not attempting to re-enumerate all documented arguments.
+ array_merge( // @phpstan-ignore argument.type
$args,
[
'number' => 1,
@@ -100,7 +104,7 @@ function find_one_user( array $args ) {
/**
* Query for and return one site.
*
- * @param array $args Query arguments.
+ * @param mixed[] $args Query arguments.
* @return mixed|null A \WP_Site object, an alternate return type if requested
* in $args, or null.
*/
@@ -108,7 +112,8 @@ function find_one_site( array $args ) {
return find_result(
\WP_Site::class,
get_sites(
- array_merge(
+ // Not attempting to re-enumerate all documented arguments.
+ array_merge( // @phpstan-ignore argument.type
$args,
[
'number' => 1,
diff --git a/src/alley/wp/find-result.php b/src/find-result.php
similarity index 61%
rename from src/alley/wp/find-result.php
rename to src/find-result.php
index 576ba8a..a1deed5 100644
--- a/src/alley/wp/find-result.php
+++ b/src/find-result.php
@@ -16,14 +16,14 @@
* Returns the first item in the given array if the item is an instance of a
* specific class or is a non-object.
*
- * @param string $class If the result plucked from $results is an object, the
- * name of the class the object must be an instance of
- * for it to be returned.
- * @param array $results The array of results from the WordPress query
- * function to retrieve the result from.
+ * @param string $class If the result plucked from $results is an object, the
+ * name of the class the object must be an instance of
+ * for it to be returned.
+ * @param mixed|mixed[] $results The array of results from the WordPress query
+ * function to retrieve the result from.
* @return mixed|null An object of the given $class; a non-object if the item
- * plucked from $results is not an object; null if no
- * result is found.
+ * plucked from $results is not an object; null if no
+ * result is found.
*/
function find_result( string $class, $results ) {
if ( ! $results || ! \is_array( $results ) ) {
diff --git a/tests/Feature/.gitkeep b/tests/Feature/.gitkeep
new file mode 100644
index 0000000..e69de29
diff --git a/tests/alley/wp/fixtures/class-noop.php b/tests/Fixtures/Noop.php
similarity index 92%
rename from tests/alley/wp/fixtures/class-noop.php
rename to tests/Fixtures/Noop.php
index 9eb21d5..5a48ead 100644
--- a/tests/alley/wp/fixtures/class-noop.php
+++ b/tests/Fixtures/Noop.php
@@ -10,7 +10,7 @@
* @package wp-find-one
*/
-namespace Alley\WP\Fixtures;
+namespace Alley\WP\Tests\Fixtures;
/**
* A class that does nothing but can be instantiated and referenced.
diff --git a/tests/Unit/.gitkeep b/tests/Unit/.gitkeep
new file mode 100644
index 0000000..e69de29
diff --git a/tests/alley/wp/test-find-one.php b/tests/Unit/FindOneTest.php
similarity index 82%
rename from tests/alley/wp/test-find-one.php
rename to tests/Unit/FindOneTest.php
index cea4959..c7cd7df 100644
--- a/tests/alley/wp/test-find-one.php
+++ b/tests/Unit/FindOneTest.php
@@ -1,6 +1,6 @@
*
@@ -10,19 +10,25 @@
* @package wp-find-one
*/
-namespace Alley\WP;
+namespace Alley\WP\Tests\Unit;
use Mantle\Testkit\Test_Case;
+use function Alley\WP\find_one_comment;
+use function Alley\WP\find_one_post;
+use function Alley\WP\find_one_site;
+use function Alley\WP\find_one_term;
+use function Alley\WP\find_one_user;
+
/**
* Unit tests for the `find_one()` functions.
*/
-final class Test_Find_One extends Test_Case {
+final class FindOneTest extends Test_Case {
/**
* Test that `find_one_post()` finds an expected \WP_Post object.
*/
public function test_find_one_post_finds_post() {
- $post_id = static::factory()->post->create(
+ $post_id = self::factory()->post->create(
[
'post_status' => 'draft',
]
@@ -43,7 +49,7 @@ public function test_find_one_post_finds_post() {
* Test that `find_one_term()` finds an expected \WP_Term object.
*/
public function test_find_one_term_finds_term() {
- $term_id = static::factory()->term->create(
+ $term_id = self::factory()->term->create(
[
'name' => 'data1',
]
@@ -66,7 +72,7 @@ public function test_find_one_term_finds_term() {
public function test_find_one_comment_finds_comment() {
$url = 'https://alley.co';
- $comment_id = static::factory()->comment->create(
+ $comment_id = self::factory()->comment->create(
[
'comment_author_url' => $url,
]
@@ -86,7 +92,7 @@ public function test_find_one_comment_finds_comment() {
* Test that `find_one_user()` finds an expected \WP_User object.
*/
public function test_find_one_user_finds_user() {
- $user_id = static::factory()->user->create(
+ $user_id = self::factory()->user->create(
[
'user_login' => 'data1',
]
@@ -113,7 +119,7 @@ public function test_find_one_site_finds_site() {
// WordPress < 5.0 doesn't ensure paths are wrapped in slashes.
$path = '/data1/';
- $blog_id = static::factory()->blog->create(
+ $blog_id = self::factory()->blog->create(
[
'path' => $path,
]
diff --git a/tests/alley/wp/test-find-result.php b/tests/Unit/FindResultTest.php
similarity index 77%
rename from tests/alley/wp/test-find-result.php
rename to tests/Unit/FindResultTest.php
index 62899eb..674f9b6 100644
--- a/tests/alley/wp/test-find-result.php
+++ b/tests/Unit/FindResultTest.php
@@ -1,6 +1,6 @@
*
@@ -10,23 +10,26 @@
* @package wp-find-one
*/
-namespace Alley\WP;
+namespace Alley\WP\Tests\Unit;
-use Alley\WP\Fixtures\Noop;
+use Alley\WP\Tests\Fixtures\Noop;
use Mantle\Testkit\Test_Case;
+use PHPUnit\Framework\Attributes\DataProvider;
+
+use function Alley\WP\find_result;
+
/**
* Unit tests for `find_result()`.
*/
-final class Test_Find_Result extends Test_Case {
+final class FindResultTest extends Test_Case {
/**
* Test that `find_result()` plucks the expected value from the given args.
*
- * @dataProvider data_find_result
- *
* @param array $args Function arguments.
* @param mixed $expected Expected return value.
*/
+ #[dataProvider( 'data_find_result' )]
public function test_find_result( $args, $expected ) { // phpcs:ignore Generic.NamingConventions.ConstructorName.OldStyle
$this->assertSame( $expected, find_result( ...$args ) );
}
@@ -36,7 +39,7 @@ public function test_find_result( $args, $expected ) { // phpcs:ignore Generic.N
*
* @return array Array of data.
*/
- public function data_find_result() {
+ public static function data_find_result() {
$noop = new Noop();
$post = 123;