HEX
Server: Apache
System: Linux pdx1-shared-a1-38 6.6.104-grsec-jammy+ #3 SMP Tue Sep 16 00:28:11 UTC 2025 x86_64
User: mmickelson (3396398)
PHP: 8.1.31
Disabled: NONE
Upload Files
File: //usr/local/wp/vendor/wp-cli/wp-cli-tests/src/Context/ThenStepDefinitions.php
<?php

namespace WP_CLI\Tests\Context;

use Behat\Gherkin\Node\PyStringNode;
use Behat\Gherkin\Node\TableNode;
use Exception;
use Requests;
use RuntimeException;

trait ThenStepDefinitions {

	use Support;

	/**
	 * Expect a specific exit code of the previous command.
	 *
	 * ```
	 * Scenario: My example scenario
	 *   Given a WP installation
	 *   When I try `wp plugin install`
	 *   Then the return code should be 1
	 * ```
	 *
	 * @access public
	 *
	 * @Then /^the return code should( not)? be (\d+)$/
	 */
	public function then_the_return_code_should_be( $not, $return_code ) {
		if (
			( ! $not && (int) $return_code !== $this->result->return_code )
			|| ( $not && (int) $return_code === $this->result->return_code )
		) {
			throw new RuntimeException( $this->result );
		}
	}

	/**
	 * Check the contents of STDOUT or STDERR.
	 *
	 * ```
	 * Scenario: My example scenario
	 *   Given an empty directory
	 *   When I run `wp core is-installed`
	 *   Then STDOUT should be empty
	 *
	 * Scenario: My other scenario
	 *   Given a WP install
	 *   When I run `wp plugin install akismet`
	 *   Then STDOUT should contain:
	 *     """
	 *     Plugin installed successfully.
	 *     """
	 *   And STDERR should be empty
	 * ```
	 *
	 * @access public
	 *
	 * @Then /^(STDOUT|STDERR) should( strictly)? (be|contain|not contain):$/
	 */
	public function then_stdout_stderr_should_contain( $stream, $strictly, $action, PyStringNode $expected ) {

		$stream = strtolower( $stream );

		$expected = $this->replace_variables( (string) $expected );

		$this->check_string( $this->result->$stream, $expected, $action, $this->result, (bool) $strictly );
	}

	/**
	 * Expect STDOUT or STDERR to be a numeric value.
	 *
	 * ```
	 * Scenario: My example scenario
	 *   Given a WP installation
	 *   When I run `wp db size --size_format=b`
	 *   Then STDOUT should be a number
	 * ```
	 *
	 * @access public
	 *
	 * @Then /^(STDOUT|STDERR) should be a number$/
	 */
	public function then_stdout_stderr_should_be_a_number( $stream ) {

		$stream = strtolower( $stream );

		$this->assert_numeric( trim( $this->result->$stream, "\n" ) );
	}

	/**
	 * Expect STDOUT or STDERR to not be a numeric value.
	 *
	 * ```
	 * Scenario: My example scenario
	 *   Given a WP installation
	 *   When I run `wp post list --format=json`
	 *   Then STDOUT should not be a number
	 * ```
	 *
	 * @access public
	 *
	 * @Then /^(STDOUT|STDERR) should not be a number$/
	 */
	public function then_stdout_stderr_should_not_be_a_number( $stream ) {

		$stream = strtolower( $stream );

		$this->assert_not_numeric( trim( $this->result->$stream, "\n" ) );
	}

	/**
	 * Expect STDOUT to be a table containing the given rows.
	 *
	 * ```
	 * Scenario: My example scenario
	 *   Given a WP installation
	 *   When I run `wp config list --fields=name,type`
	 *   Then STDOUT should be a table containing rows:
	 *     | name    | type     |
	 *     | DB_NAME | constant |
	 *     | DB_USER | constant |
	 * ```
	 *
	 * @access public
	 *
	 * @Then /^STDOUT should be a table containing rows:$/
	 */
	public function then_stdout_should_be_a_table_containing_rows( TableNode $expected ) {
		$output      = $this->result->stdout;
		$actual_rows = explode( "\n", rtrim( $output, "\n" ) );

		$expected_rows = array();
		foreach ( $expected->getRows() as $row ) {
			$expected_rows[] = $this->replace_variables( implode( "\t", $row ) );
		}

		$this->compare_tables( $expected_rows, $actual_rows, $output );
	}

	/**
	 * Expect STDOUT to end with a table containing the given rows.
	 *
	 * Useful when the table is preceded by some other output.
	 *
	 * ```
	 * Scenario: My example scenario
	 *   Given a WP installation
	 *   When I run `wp search-replace foo bar --report-changed-only`
	 *   Then STDOUT should contain:
	 *     """
	 *     Success: Made 3 replacements.
	 *     """
	 *   And STDOUT should end with a table containing rows:
	 *     | Table       | Column       | Replacements | Type |
	 *     | wp_options  | option_value | 1            | PHP  |
	 *     | wp_postmeta | meta_value   | 1            | SQL  |
	 *     | wp_posts    | post_title   | 1            | SQL  |
	 * ```
	 *
	 * @access public
	 *
	 * @Then /^STDOUT should end with a table containing rows:$/
	 */
	public function then_stdout_should_end_with_a_table_containing_rows( TableNode $expected ) {
		$output      = $this->result->stdout;
		$actual_rows = explode( "\n", rtrim( $output, "\n" ) );

		$expected_rows = array();
		foreach ( $expected->getRows() as $row ) {
			$expected_rows[] = $this->replace_variables( implode( "\t", $row ) );
		}

		$start = array_search( $expected_rows[0], $actual_rows, true );

		if ( false === $start ) {
			throw new Exception( $this->result );
		}

		$this->compare_tables( $expected_rows, array_slice( $actual_rows, $start ), $output );
	}

	/**
	 * Expect valid JSON output in STDOUT.
	 *
	 * ```
	 * Scenario: My example scenario
	 *   When I run `wp post meta get 1 meta-key --format=json`
	 *   Then STDOUT should be JSON containing:
	 *     """
	 *     {
	 *     "foo": "baz"
	 *     }
	 *   """
	 * ```
	 *
	 * @access public
	 *
	 * @Then /^STDOUT should be JSON containing:$/
	 */
	public function then_stdout_should_be_json_containing( PyStringNode $expected ) {
		$output   = $this->result->stdout;
		$expected = $this->replace_variables( (string) $expected );

		if ( ! $this->check_that_json_string_contains_json_string( $output, $expected ) ) {
			throw new Exception( $this->result );
		}
	}

	/**
	 * Expect valid JSON array output in STDOUT.
	 *
	 * Errors when some items are missing from the expected array.
	 *
	 * ```
	 * Scenario: My example scenario
	 *   When I run `wp plugin list --field=name --format=json`
	 *   Then STDOUT should be a JSON array containing:
	 *     """
	 *     ["akismet", "hello-dolly"]
	 *     """
	 * ```
	 *
	 * @access public
	 *
	 * @Then /^STDOUT should be a JSON array containing:$/
	 */
	public function then_stdout_should_be_a_json_array_containing( PyStringNode $expected ) {
		$output   = $this->result->stdout;
		$expected = $this->replace_variables( (string) $expected );

		$actual_values   = json_decode( $output );
		$expected_values = json_decode( $expected );

		$missing = array_diff( $expected_values, $actual_values );
		if ( ! empty( $missing ) ) {
			throw new Exception( $this->result );
		}
	}

	/**
	 * Expect STDOUT to be CSV containing certain values.
	 *
	 * ```
	 * Scenario: My example scenario
	 *   When I run `wp term list post_tag --fields=name,slug --format=csv`
	 *   Then STDOUT should be CSV containing:
	 *     | name      | slug |
	 *     | Test term | test |
	 * ```
	 *
	 * @access public
	 *
	 * @Then /^STDOUT should be CSV containing:$/
	 */
	public function then_stdout_should_be_csv_containing( TableNode $expected ) {
		$output = $this->result->stdout;

		$expected_rows = $expected->getRows();
		foreach ( $expected as &$row ) {
			foreach ( $row as &$value ) {
				$value = $this->replace_variables( $value );
			}
		}

		if ( ! $this->check_that_csv_string_contains_values( $output, $expected_rows ) ) {
			throw new Exception( $this->result );
		}
	}

	/**
	 * Expect STDOUT to be YAML containig certain content.
	 *
	 * ```
	 * Scenario: My example scenario
	 *   When I run `wp cli alias list`
	 *   Then STDOUT should be YAML containing:
	 *     """
	 *     @all: Run command against every registered alias.
	 *     @foo:
	 *       path: {TEST_DIR}/foo
	 *     """
	 * ```
	 *
	 * @access public
	 *
	 * @Then /^STDOUT should be YAML containing:$/
	 */
	public function then_stdout_should_be_yaml_containing( PyStringNode $expected ) {
		$output   = $this->result->stdout;
		$expected = $this->replace_variables( (string) $expected );

		if ( ! $this->check_that_yaml_string_contains_yaml_string( $output, $expected ) ) {
			throw new Exception( $this->result );
		}
	}

	/**
	 * Expect STDOUT or STDERR to be empty.
	 *
	 * ```
	 * Scenario: My other scenario
	 *   Given a WP install
	 *   When I run `wp plugin install akismet`
	 *   Then STDERR should be empty
	 * ```
	 *
	 * @access public
	 *
	 * @Then /^(STDOUT|STDERR) should be empty$/
	 */
	public function then_stdout_stderr_should_be_empty( $stream ) {

		$stream = strtolower( $stream );

		if ( ! empty( $this->result->$stream ) ) {
			throw new Exception( $this->result );
		}
	}

	/**
	 * Expect STDOUT or STDERR not to be empty.
	 *
	 * ```
	 * Scenario: My example scenario
	 *   When I run `wp user create examplejane jane@example.com`
	 *   Then STDOUT should not be empty
	 * ```
	 *
	 * @access public
	 *
	 * @Then /^(STDOUT|STDERR) should not be empty$/
	 */
	public function then_stdout_stderr_should_not_be_empty( $stream ) {

		$stream = strtolower( $stream );

		if ( '' === rtrim( $this->result->$stream, "\n" ) ) {
			throw new Exception( $this->result );
		}
	}

	/**
	 * Expect STDOUT or STDERR to be a version string comparing to the given version.
	 *
	 * ```
	 * Scenario: My example scenario
	 *   Given a WP install
	 *   When I run `wp core version
	 *   Then STDOUT should be a version string >= 6.8
	 * ```
	 *
	 * @access public
	 *
	 * @Then /^(STDOUT|STDERR) should be a version string (<|<=|>|>=|==|=|!=|<>) ([+\w.{}-]+)$/
	 */
	public function then_stdout_stderr_should_be_a_specific_version_string( $stream, $operator, $goal_ver ) {
		$goal_ver = $this->replace_variables( $goal_ver );
		$stream   = strtolower( $stream );
		if ( false === version_compare( trim( $this->result->$stream, "\n" ), $goal_ver, $operator ) ) {
			throw new Exception( $this->result );
		}
	}

	/**
	 * Expect a certain file or directory to (not) exist or (not) contain certain contents.
	 *
	 * ```
	 * Scenario: My example scenario
	 *   When I run `wp core download`
	 *   Then the wp-settings.php file should exist
	 *   And the wp-content directory should exist
	 *   And the {RUN_DIR} directory should contain:
	 *     """
	 *     index.php
	 *     license.txt
	 *     """
	 *   And the wp-config.php file should contain:
	 *     """
	 *     That's all, stop editing! Happy publishing.
	 *     """
	 * ```
	 *
	 * @access public
	 *
	 * @Then /^the (.+) (file|directory) should( strictly)? (exist|not exist|be:|contain:|not contain:)$/
	 */
	public function then_a_specific_file_folder_should_exist( $path, $type, $strictly, $action, $expected = null ) {
		$path = $this->replace_variables( $path );

		// If it's a relative path, make it relative to the current test dir.
		if ( '/' !== $path[0] ) {
			$path = $this->variables['RUN_DIR'] . "/$path";
		}

		$exists = static function ( $path ) use ( $type ) {
			// Clear the stat cache for the path first to avoid
			// potentially inaccurate results when files change outside of PHP.
			// See https://www.php.net/manual/en/function.clearstatcache.php
			clearstatcache( false, $path );

			if ( 'directory' === $type ) {
				return is_dir( $path );
			}

			return file_exists( $path );
		};

		switch ( $action ) {
			case 'exist':
				if ( ! $exists( $path ) ) {
					throw new Exception( "$path doesn't exist." );
				}
				break;
			case 'not exist':
				if ( $exists( $path ) ) {
					throw new Exception( "$path exists." );
				}
				break;
			default:
				if ( ! $exists( $path ) ) {
					throw new Exception( "$path doesn't exist." );
				}
				$action   = substr( $action, 0, -1 );
				$expected = $this->replace_variables( (string) $expected );
				if ( 'file' === $type ) {
					$contents = file_get_contents( $path );
				} elseif ( 'directory' === $type ) {
					$files = glob( rtrim( $path, '/' ) . '/*' );
					foreach ( $files as &$file ) {
						$file = str_replace( $path . '/', '', $file );
					}
					$contents = implode( PHP_EOL, $files );
				}
				$this->check_string( $contents, $expected, $action, false, (bool) $strictly );
		}
	}

	/**
	 * Match file contents against a regex.
	 *
	 * ```
	 * Scenario: My example scenario
	 *   When I run `wp scaffold plugin hello-world`
	 *   Then the contents of the wp-content/plugins/hello-world/languages/hello-world.pot file should match /X-Generator:\s/
	 * ```
	 *
	 * @access public
	 *
	 * @Then /^the contents of the (.+) file should( not)? match (((\/.+\/)|(#.+#))([a-z]+)?)$/
	 */
	public function then_the_contents_of_a_specific_file_should_match( $path, $not, $expected ) {
		$path     = $this->replace_variables( $path );
		$expected = $this->replace_variables( $expected );

		// If it's a relative path, make it relative to the current test dir.
		if ( '/' !== $path[0] ) {
			$path = $this->variables['RUN_DIR'] . "/$path";
		}
		$contents = file_get_contents( $path );
		if ( $not ) {
			$this->assert_not_regex( $expected, $contents );
		} else {
			$this->assert_regex( $expected, $contents );
		}
	}

	/**
	 * Match STDOUT or STDERR against a regex.
	 *
	 * ```
	 * Scenario: My example scenario
	 *   When I run `wp dist-archive wp-content/plugins/hello-world`
	 *   Then STDOUT should match /^Success: Created hello-world.0.1.0.zip \(Size: \d+(?:\.\d*)? [a-zA-Z]{1,3}\)$/
	 * ```
	 *
	 * @access public
	 *
	 * @Then /^(STDOUT|STDERR) should( not)? match (((\/.+\/)|(#.+#))([a-z]+)?)$/
	 */
	public function then_stdout_stderr_should_match_a_string( $stream, $not, $expected ) {
		$expected = $this->replace_variables( $expected );

		$stream = strtolower( $stream );
		if ( $not ) {
			$this->assert_not_regex( $expected, $this->result->$stream );
		} else {
			$this->assert_regex( $expected, $this->result->$stream );
		}
	}

	/**
	 * Expect an email to be sent (or not).
	 *
	 * ```
	 * Scenario: My example scenario
	 *   When I run `wp user reset-password 1`
	 *   Then an email should be sent
	 * ```
	 *
	 * @access public
	 *
	 * @Then /^an email should (be sent|not be sent)$/
	 */
	public function then_an_email_should_be_sent( $expected ) {
		if ( 'be sent' === $expected ) {
			$this->assert_not_equals( 0, $this->email_sends );
		} elseif ( 'not be sent' === $expected ) {
			$this->assert_equals( 0, $this->email_sends );
		} else {
			throw new Exception( 'Invalid expectation' );
		}
	}

	/**
	 * Expect the HTTP status code for visiting `http://localhost:8080`.
	 *
	 * ```
	 * Scenario: My example scenario
	 *   Given a WP installation with Composer
	 *   And a PHP built-in web server to serve 'WordPress'
	 *   Then the HTTP status code should be 200
	 * ```
	 *
	 * @access public
	 *
	 * @Then the HTTP status code should be :code
	 */
	public function then_the_http_status_code_should_be( $return_code ) {
		$response = Requests::request( 'http://localhost:8080' );
		$this->assert_equals( $return_code, $response->status_code );
	}
}