| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299 |
- <?php
- declare(strict_types=1);
- /**
- * CakePHP(tm) : Rapid Development Framework (https://cakephp.org)
- * Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
- *
- * Licensed under The MIT License
- * For full copyright and license information, please see the LICENSE.txt
- * Redistributions of files must retain the above copyright notice
- *
- * @copyright Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
- * @since 3.7.0
- * @license https://opensource.org/licenses/mit-license.php MIT License
- */
- namespace Cake\TestSuite;
- use Cake\Controller\Controller;
- use Cake\Core\Configure;
- use Cake\Database\Exception as DatabaseException;
- use Cake\Error\ExceptionRenderer;
- use Cake\Event\EventInterface;
- use Cake\Form\FormProtector;
- use Cake\Http\Session;
- use Cake\Routing\Router;
- use Cake\TestSuite\Constraint\Response\BodyContains;
- use Cake\TestSuite\Constraint\Response\BodyEmpty;
- use Cake\TestSuite\Constraint\Response\BodyEquals;
- use Cake\TestSuite\Constraint\Response\BodyNotContains;
- use Cake\TestSuite\Constraint\Response\BodyNotEmpty;
- use Cake\TestSuite\Constraint\Response\BodyNotEquals;
- use Cake\TestSuite\Constraint\Response\BodyNotRegExp;
- use Cake\TestSuite\Constraint\Response\BodyRegExp;
- use Cake\TestSuite\Constraint\Response\ContentType;
- use Cake\TestSuite\Constraint\Response\CookieEncryptedEquals;
- use Cake\TestSuite\Constraint\Response\CookieEquals;
- use Cake\TestSuite\Constraint\Response\CookieNotSet;
- use Cake\TestSuite\Constraint\Response\CookieSet;
- use Cake\TestSuite\Constraint\Response\FileSent;
- use Cake\TestSuite\Constraint\Response\FileSentAs;
- use Cake\TestSuite\Constraint\Response\HeaderContains;
- use Cake\TestSuite\Constraint\Response\HeaderEquals;
- use Cake\TestSuite\Constraint\Response\HeaderNotContains;
- use Cake\TestSuite\Constraint\Response\HeaderNotSet;
- use Cake\TestSuite\Constraint\Response\HeaderSet;
- use Cake\TestSuite\Constraint\Response\StatusCode;
- use Cake\TestSuite\Constraint\Response\StatusError;
- use Cake\TestSuite\Constraint\Response\StatusFailure;
- use Cake\TestSuite\Constraint\Response\StatusOk;
- use Cake\TestSuite\Constraint\Response\StatusSuccess;
- use Cake\TestSuite\Constraint\Session\FlashParamEquals;
- use Cake\TestSuite\Constraint\Session\SessionEquals;
- use Cake\TestSuite\Constraint\View\LayoutFileEquals;
- use Cake\TestSuite\Constraint\View\TemplateFileEquals;
- use Cake\TestSuite\Stub\TestExceptionRenderer;
- use Cake\Utility\CookieCryptTrait;
- use Cake\Utility\Hash;
- use Cake\Utility\Security;
- use Cake\Utility\Text;
- use Exception;
- use LogicException;
- use PHPUnit\Framework\Error\Error as PhpUnitError;
- use Throwable;
- use Zend\Diactoros\Uri;
- /**
- * A trait intended to make integration tests of your controllers easier.
- *
- * This test class provides a number of helper methods and features
- * that make dispatching requests and checking their responses simpler.
- * It favours full integration tests over mock objects as you can test
- * more of your code easily and avoid some of the maintenance pitfalls
- * that mock objects create.
- */
- trait IntegrationTestTrait
- {
- use CookieCryptTrait;
- /**
- * The customized application class name.
- *
- * @var string|null
- */
- protected $_appClass;
- /**
- * The customized application constructor arguments.
- *
- * @var array|null
- */
- protected $_appArgs;
- /**
- * The data used to build the next request.
- *
- * @var array
- */
- protected $_request = [];
- /**
- * The response for the most recent request.
- *
- * @var \Psr\Http\Message\ResponseInterface
- */
- protected $_response;
- /**
- * The exception being thrown if the case.
- *
- * @var \Throwable|null
- */
- protected $_exception;
- /**
- * Session data to use in the next request.
- *
- * @var array
- */
- protected $_session = [];
- /**
- * Cookie data to use in the next request.
- *
- * @var array
- */
- protected $_cookie = [];
- /**
- * The controller used in the last request.
- *
- * @var \Cake\Controller\Controller
- */
- protected $_controller;
- /**
- * The last rendered view
- *
- * @var string
- */
- protected $_viewName;
- /**
- * The last rendered layout
- *
- * @var string
- */
- protected $_layoutName;
- /**
- * The session instance from the last request
- *
- * @var \Cake\Http\Session
- */
- protected $_requestSession;
- /**
- * Boolean flag for whether or not the request should have
- * a SecurityComponent token added.
- *
- * @var bool
- */
- protected $_securityToken = false;
- /**
- * Boolean flag for whether or not the request should have
- * a CSRF token added.
- *
- * @var bool
- */
- protected $_csrfToken = false;
- /**
- * Boolean flag for whether or not the request should re-store
- * flash messages
- *
- * @var bool
- */
- protected $_retainFlashMessages = false;
- /**
- * Stored flash messages before render
- *
- * @var array|null
- */
- protected $_flashMessages;
- /**
- *
- * @var string|null
- */
- protected $_cookieEncryptionKey;
- /**
- * List of fields that are excluded from field validation.
- *
- * @var string[]
- */
- protected $_unlockedFields = [];
- /**
- * Clears the state used for requests.
- *
- * @after
- * @return void
- * @psalm-suppress PossiblyNullPropertyAssignmentValue
- */
- public function cleanup(): void
- {
- $this->_request = [];
- $this->_session = [];
- $this->_cookie = [];
- $this->_response = null;
- $this->_exception = null;
- $this->_controller = null;
- $this->_viewName = null;
- $this->_layoutName = null;
- $this->_requestSession = null;
- $this->_appClass = null;
- $this->_appArgs = null;
- $this->_securityToken = false;
- $this->_csrfToken = false;
- $this->_retainFlashMessages = false;
- }
- /**
- * Configure the application class to use in integration tests.
- *
- * @param string $class The application class name.
- * @param array|null $constructorArgs The constructor arguments for your application class.
- * @return void
- */
- public function configApplication(string $class, ?array $constructorArgs): void
- {
- $this->_appClass = $class;
- $this->_appArgs = $constructorArgs;
- }
- /**
- * Calling this method will enable a SecurityComponent
- * compatible token to be added to request data. This
- * lets you easily test actions protected by SecurityComponent.
- *
- * @return void
- */
- public function enableSecurityToken(): void
- {
- $this->_securityToken = true;
- }
- /**
- * Set list of fields that are excluded from field validation.
- *
- * @param string[] $unlockedFields List of fields that are excluded from field validation.
- * @return void
- */
- public function setUnlockedFields(array $unlockedFields = []): void
- {
- $this->_unlockedFields = $unlockedFields;
- }
- /**
- * Calling this method will add a CSRF token to the request.
- *
- * Both the POST data and cookie will be populated when this option
- * is enabled. The default parameter names will be used.
- *
- * @return void
- */
- public function enableCsrfToken(): void
- {
- $this->_csrfToken = true;
- }
- /**
- * Calling this method will re-store flash messages into the test session
- * after being removed by the FlashHelper
- *
- * @return void
- */
- public function enableRetainFlashMessages(): void
- {
- $this->_retainFlashMessages = true;
- }
- /**
- * Configures the data for the *next* request.
- *
- * This data is cleared in the tearDown() method.
- *
- * You can call this method multiple times to append into
- * the current state.
- *
- * @param array $data The request data to use.
- * @return void
- */
- public function configRequest(array $data): void
- {
- $this->_request = $data + $this->_request;
- }
- /**
- * Sets session data.
- *
- * This method lets you configure the session data
- * you want to be used for requests that follow. The session
- * state is reset in each tearDown().
- *
- * You can call this method multiple times to append into
- * the current state.
- *
- * @param array $data The session data to use.
- * @return void
- */
- public function session(array $data): void
- {
- $this->_session = $data + $this->_session;
- }
- /**
- * Sets a request cookie for future requests.
- *
- * This method lets you configure the session data
- * you want to be used for requests that follow. The session
- * state is reset in each tearDown().
- *
- * You can call this method multiple times to append into
- * the current state.
- *
- * @param string $name The cookie name to use.
- * @param mixed $value The value of the cookie.
- * @return void
- */
- public function cookie(string $name, $value): void
- {
- $this->_cookie[$name] = $value;
- }
- /**
- * Returns the encryption key to be used.
- *
- * @return string
- */
- protected function _getCookieEncryptionKey(): string
- {
- if (isset($this->_cookieEncryptionKey)) {
- return $this->_cookieEncryptionKey;
- }
- return Security::getSalt();
- }
- /**
- * Sets a encrypted request cookie for future requests.
- *
- * The difference from cookie() is this encrypts the cookie
- * value like the CookieComponent.
- *
- * @param string $name The cookie name to use.
- * @param mixed $value The value of the cookie.
- * @param string|false $encrypt Encryption mode to use.
- * @param string|null $key Encryption key used. Defaults
- * to Security.salt.
- * @return void
- * @see \Cake\Utility\CookieCryptTrait::_encrypt()
- */
- public function cookieEncrypted(string $name, $value, $encrypt = 'aes', $key = null): void
- {
- $this->_cookieEncryptionKey = $key;
- $this->_cookie[$name] = $this->_encrypt($value, $encrypt);
- }
- /**
- * Performs a GET request using the current request data.
- *
- * The response of the dispatched request will be stored as
- * a property. You can use various assert methods to check the
- * response.
- *
- * @param string|array $url The URL to request.
- * @return void
- * @throws \PHPUnit\Framework\Error\Error|\Throwable
- */
- public function get($url): void
- {
- $this->_sendRequest($url, 'GET');
- }
- /**
- * Performs a POST request using the current request data.
- *
- * The response of the dispatched request will be stored as
- * a property. You can use various assert methods to check the
- * response.
- *
- * @param string|array $url The URL to request.
- * @param string|array $data The data for the request.
- * @return void
- * @throws \PHPUnit\Framework\Error\Error|\Throwable
- */
- public function post($url, $data = []): void
- {
- $this->_sendRequest($url, 'POST', $data);
- }
- /**
- * Performs a PATCH request using the current request data.
- *
- * The response of the dispatched request will be stored as
- * a property. You can use various assert methods to check the
- * response.
- *
- * @param string|array $url The URL to request.
- * @param string|array $data The data for the request.
- * @return void
- * @throws \PHPUnit\Framework\Error\Error|\Throwable
- */
- public function patch($url, $data = []): void
- {
- $this->_sendRequest($url, 'PATCH', $data);
- }
- /**
- * Performs a PUT request using the current request data.
- *
- * The response of the dispatched request will be stored as
- * a property. You can use various assert methods to check the
- * response.
- *
- * @param string|array $url The URL to request.
- * @param string|array $data The data for the request.
- * @return void
- * @throws \PHPUnit\Framework\Error\Error|\Throwable
- */
- public function put($url, $data = []): void
- {
- $this->_sendRequest($url, 'PUT', $data);
- }
- /**
- * Performs a DELETE request using the current request data.
- *
- * The response of the dispatched request will be stored as
- * a property. You can use various assert methods to check the
- * response.
- *
- * @param string|array $url The URL to request.
- * @return void
- * @throws \PHPUnit\Framework\Error\Error|\Throwable
- */
- public function delete($url): void
- {
- $this->_sendRequest($url, 'DELETE');
- }
- /**
- * Performs a HEAD request using the current request data.
- *
- * The response of the dispatched request will be stored as
- * a property. You can use various assert methods to check the
- * response.
- *
- * @param string|array $url The URL to request.
- * @return void
- * @throws \PHPUnit\Framework\Error\Error|\Throwable
- */
- public function head($url): void
- {
- $this->_sendRequest($url, 'HEAD');
- }
- /**
- * Performs an OPTIONS request using the current request data.
- *
- * The response of the dispatched request will be stored as
- * a property. You can use various assert methods to check the
- * response.
- *
- * @param string|array $url The URL to request.
- * @return void
- * @throws \PHPUnit\Framework\Error\Error|\Throwable
- */
- public function options($url): void
- {
- $this->_sendRequest($url, 'OPTIONS');
- }
- /**
- * Creates and send the request into a Dispatcher instance.
- *
- * Receives and stores the response for future inspection.
- *
- * @param string|array $url The URL
- * @param string $method The HTTP method
- * @param string|array $data The request data.
- * @return void
- * @throws \PHPUnit\Framework\Error\Error|\Throwable
- */
- protected function _sendRequest($url, $method, $data = []): void
- {
- $dispatcher = $this->_makeDispatcher();
- $url = $dispatcher->resolveUrl($url);
- try {
- $request = $this->_buildRequest($url, $method, $data);
- $response = $dispatcher->execute($request);
- $this->_requestSession = $request['session'];
- if ($this->_retainFlashMessages && $this->_flashMessages) {
- $this->_requestSession->write('Flash', $this->_flashMessages);
- }
- $this->_response = $response;
- } catch (PhpUnitError $e) {
- throw $e;
- } catch (DatabaseException $e) {
- throw $e;
- } catch (LogicException $e) {
- throw $e;
- } catch (Throwable $e) {
- $this->_exception = $e;
- // Simulate the global exception handler being invoked.
- $this->_handleError($e);
- }
- }
- /**
- * Get the correct dispatcher instance.
- *
- * @return \Cake\TestSuite\MiddlewareDispatcher A dispatcher instance
- */
- protected function _makeDispatcher(): MiddlewareDispatcher
- {
- return new MiddlewareDispatcher($this, $this->_appClass, $this->_appArgs);
- }
- /**
- * Adds additional event spies to the controller/view event manager.
- *
- * @param \Cake\Event\EventInterface $event A dispatcher event.
- * @param \Cake\Controller\Controller|null $controller Controller instance.
- * @return void
- */
- public function controllerSpy(EventInterface $event, ?Controller $controller = null): void
- {
- if (!$controller) {
- /** @var \Cake\Controller\Controller $controller */
- $controller = $event->getSubject();
- }
- $this->_controller = $controller;
- $events = $controller->getEventManager();
- $events->on('View.beforeRender', function ($event, $viewFile) use ($controller): void {
- if (!$this->_viewName) {
- $this->_viewName = $viewFile;
- }
- if ($this->_retainFlashMessages) {
- /** @var array|null $this->_flashMessages */
- $this->_flashMessages = $controller->getRequest()->getSession()->read('Flash');
- }
- });
- $events->on('View.beforeLayout', function ($event, $viewFile): void {
- $this->_layoutName = $viewFile;
- });
- }
- /**
- * Attempts to render an error response for a given exception.
- *
- * This method will attempt to use the configured exception renderer.
- * If that class does not exist, the built-in renderer will be used.
- *
- * @param \Throwable $exception Exception to handle.
- * @return void
- */
- protected function _handleError(Throwable $exception): void
- {
- $class = Configure::read('Error.exceptionRenderer');
- if (empty($class) || !class_exists($class)) {
- $class = ExceptionRenderer::class;
- }
- /** @var \Cake\Error\ExceptionRenderer $instance */
- $instance = new $class($exception);
- $this->_response = $instance->render();
- }
- /**
- * Creates a request object with the configured options and parameters.
- *
- * @param string $url The URL
- * @param string $method The HTTP method
- * @param string|array $data The request data.
- * @return array The request context
- */
- protected function _buildRequest(string $url, $method, $data = []): array
- {
- $sessionConfig = (array)Configure::read('Session') + [
- 'defaults' => 'php',
- ];
- $session = Session::create($sessionConfig);
- $session->write($this->_session);
- [$url, $query, $hostInfo] = $this->_url($url);
- $tokenUrl = $url;
- if ($query) {
- $tokenUrl .= '?' . $query;
- }
- parse_str($query, $queryData);
- $props = [
- 'url' => $url,
- 'session' => $session,
- 'query' => $queryData,
- 'files' => [],
- ];
- if (is_string($data)) {
- $props['input'] = $data;
- } else {
- $data = $this->_addTokens($tokenUrl, $data);
- $props['post'] = $this->_castToString($data);
- }
- $props['cookies'] = $this->_cookie;
- $env = [
- 'REQUEST_METHOD' => $method,
- 'QUERY_STRING' => $query,
- 'REQUEST_URI' => $url,
- ];
- if (!empty($hostInfo['ssl'])) {
- $env['HTTPS'] = 'on';
- }
- if (isset($hostInfo['host'])) {
- $env['HTTP_HOST'] = $hostInfo['host'];
- }
- if (isset($this->_request['headers'])) {
- foreach ($this->_request['headers'] as $k => $v) {
- $name = strtoupper(str_replace('-', '_', $k));
- if (!in_array($name, ['CONTENT_LENGTH', 'CONTENT_TYPE'], true)) {
- $name = 'HTTP_' . $name;
- }
- $env[$name] = $v;
- }
- unset($this->_request['headers']);
- }
- $props['environment'] = $env;
- $props = Hash::merge($props, $this->_request);
- return $props;
- }
- /**
- * Add the CSRF and Security Component tokens if necessary.
- *
- * @param string $url The URL the form is being submitted on.
- * @param array $data The request body data.
- * @return array The request body with tokens added.
- */
- protected function _addTokens(string $url, array $data): array
- {
- if ($this->_securityToken === true) {
- $fields = array_diff_key($data, array_flip($this->_unlockedFields));
- $keys = array_map(function ($field) {
- return preg_replace('/(\.\d+)+$/', '', $field);
- }, array_keys(Hash::flatten($fields)));
- $formProtector = new FormProtector($url, 'cli', ['unlockedFields' => $this->_unlockedFields]);
- foreach ($keys as $field) {
- /** @psalm-suppress PossiblyNullArgument */
- $formProtector->addField($field);
- }
- $tokenData = $formProtector->buildTokenData();
- $data['_Token'] = $tokenData;
- $data['_Token']['debug'] = 'FormProtector debug data would be added here';
- }
- if ($this->_csrfToken === true) {
- if (!isset($this->_cookie['csrfToken'])) {
- $this->_cookie['csrfToken'] = Text::uuid();
- }
- if (!isset($data['_csrfToken'])) {
- $data['_csrfToken'] = $this->_cookie['csrfToken'];
- }
- }
- return $data;
- }
- /**
- * Recursively casts all data to string as that is how data would be POSTed in
- * the real world
- *
- * @param array $data POST data
- * @return array
- */
- protected function _castToString(array $data): array
- {
- foreach ($data as $key => $value) {
- if (is_scalar($value)) {
- $data[$key] = $value === false ? '0' : (string)$value;
- continue;
- }
- if (is_array($value)) {
- $looksLikeFile = isset($value['error'], $value['tmp_name'], $value['size']);
- if ($looksLikeFile) {
- continue;
- }
- $data[$key] = $this->_castToString($value);
- }
- }
- return $data;
- }
- /**
- * Creates a valid request url and parameter array more like Request::_url()
- *
- * @param string $url The URL
- * @return array Qualified URL, the query parameters, and host data
- */
- protected function _url(string $url): array
- {
- $uri = new Uri($url);
- $path = $uri->getPath();
- $query = $uri->getQuery();
- $hostData = [];
- if ($uri->getHost()) {
- $hostData['host'] = $uri->getHost();
- }
- if ($uri->getScheme()) {
- $hostData['ssl'] = $uri->getScheme() === 'https';
- }
- return [$path, $query, $hostData];
- }
- /**
- * Get the response body as string
- *
- * @return string The response body.
- */
- protected function _getBodyAsString(): string
- {
- if (!$this->_response) {
- $this->fail('No response set, cannot assert content.');
- }
- return (string)$this->_response->getBody();
- }
- /**
- * Fetches a view variable by name.
- *
- * If the view variable does not exist, null will be returned.
- *
- * @param string $name The view variable to get.
- * @return mixed The view variable if set.
- */
- public function viewVariable(string $name)
- {
- return $this->_controller ? $this->_controller->viewBuilder()->getVar($name) : null;
- }
- /**
- * Asserts that the response status code is in the 2xx range.
- *
- * @param string $message Custom message for failure.
- * @return void
- */
- public function assertResponseOk(string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat(null, new StatusOk($this->_response), $verboseMessage);
- }
- /**
- * Asserts that the response status code is in the 2xx/3xx range.
- *
- * @param string $message Custom message for failure.
- * @return void
- */
- public function assertResponseSuccess(string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat(null, new StatusSuccess($this->_response), $verboseMessage);
- }
- /**
- * Asserts that the response status code is in the 4xx range.
- *
- * @param string $message Custom message for failure.
- * @return void
- */
- public function assertResponseError(string $message = ''): void
- {
- $this->assertThat(null, new StatusError($this->_response), $message);
- }
- /**
- * Asserts that the response status code is in the 5xx range.
- *
- * @param string $message Custom message for failure.
- * @return void
- */
- public function assertResponseFailure(string $message = ''): void
- {
- $this->assertThat(null, new StatusFailure($this->_response), $message);
- }
- /**
- * Asserts a specific response status code.
- *
- * @param int $code Status code to assert.
- * @param string $message Custom message for failure.
- * @return void
- */
- public function assertResponseCode(int $code, string $message = ''): void
- {
- $this->assertThat($code, new StatusCode($this->_response), $message);
- }
- /**
- * Asserts that the Location header is correct. Comparison is made against a full URL.
- *
- * @param string|array|null $url The URL you expected the client to go to. This
- * can either be a string URL or an array compatible with Router::url(). Use null to
- * simply check for the existence of this header.
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertRedirect($url = null, $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat(null, new HeaderSet($this->_response, 'Location'), $verboseMessage);
- if ($url) {
- $this->assertThat(
- Router::url($url, true),
- new HeaderEquals($this->_response, 'Location'),
- $verboseMessage
- );
- }
- }
- /**
- * Asserts that the Location header is correct. Comparison is made against exactly the URL provided.
- *
- * @param string|array|null $url The URL you expected the client to go to. This
- * can either be a string URL or an array compatible with Router::url(). Use null to
- * simply check for the existence of this header.
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertRedirectEquals($url = null, $message = '')
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat(null, new HeaderSet($this->_response, 'Location'), $verboseMessage);
- if ($url) {
- $this->assertThat(Router::url($url), new HeaderEquals($this->_response, 'Location'), $verboseMessage);
- }
- }
- /**
- * Asserts that the Location header contains a substring
- *
- * @param string $url The URL you expected the client to go to.
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertRedirectContains(string $url, string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat(null, new HeaderSet($this->_response, 'Location'), $verboseMessage);
- $this->assertThat($url, new HeaderContains($this->_response, 'Location'), $verboseMessage);
- }
- /**
- * Asserts that the Location header does not contain a substring
- *
- * @param string $url The URL you expected the client to go to.
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertRedirectNotContains(string $url, string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat(null, new HeaderSet($this->_response, 'Location'), $verboseMessage);
- $this->assertThat($url, new HeaderNotContains($this->_response, 'Location'), $verboseMessage);
- }
- /**
- * Asserts that the Location header is not set.
- *
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertNoRedirect(string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat(null, new HeaderNotSet($this->_response, 'Location'), $verboseMessage);
- }
- /**
- * Asserts response headers
- *
- * @param string $header The header to check
- * @param string $content The content to check for.
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertHeader(string $header, string $content, string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat(null, new HeaderSet($this->_response, $header), $verboseMessage);
- $this->assertThat($content, new HeaderEquals($this->_response, $header), $verboseMessage);
- }
- /**
- * Asserts response header contains a string
- *
- * @param string $header The header to check
- * @param string $content The content to check for.
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertHeaderContains(string $header, string $content, string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat(null, new HeaderSet($this->_response, $header), $verboseMessage);
- $this->assertThat($content, new HeaderContains($this->_response, $header), $verboseMessage);
- }
- /**
- * Asserts response header does not contain a string
- *
- * @param string $header The header to check
- * @param string $content The content to check for.
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertHeaderNotContains(string $header, string $content, string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat(null, new HeaderSet($this->_response, $header), $verboseMessage);
- $this->assertThat($content, new HeaderNotContains($this->_response, $header), $verboseMessage);
- }
- /**
- * Asserts content type
- *
- * @param string $type The content-type to check for.
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertContentType(string $type, string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat($type, new ContentType($this->_response), $verboseMessage);
- }
- /**
- * Asserts content in the response body equals.
- *
- * @param mixed $content The content to check for.
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertResponseEquals($content, $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat($content, new BodyEquals($this->_response), $verboseMessage);
- }
- /**
- * Asserts content in the response body not equals.
- *
- * @param mixed $content The content to check for.
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertResponseNotEquals($content, $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat($content, new BodyNotEquals($this->_response), $verboseMessage);
- }
- /**
- * Asserts content exists in the response body.
- *
- * @param string $content The content to check for.
- * @param string $message The failure message that will be appended to the generated message.
- * @param bool $ignoreCase A flag to check whether we should ignore case or not.
- * @return void
- */
- public function assertResponseContains(string $content, string $message = '', bool $ignoreCase = false): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat($content, new BodyContains($this->_response, $ignoreCase), $verboseMessage);
- }
- /**
- * Asserts content does not exist in the response body.
- *
- * @param string $content The content to check for.
- * @param string $message The failure message that will be appended to the generated message.
- * @param bool $ignoreCase A flag to check whether we should ignore case or not.
- * @return void
- */
- public function assertResponseNotContains(string $content, string $message = '', bool $ignoreCase = false): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat($content, new BodyNotContains($this->_response, $ignoreCase), $verboseMessage);
- }
- /**
- * Asserts that the response body matches a given regular expression.
- *
- * @param string $pattern The pattern to compare against.
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertResponseRegExp(string $pattern, string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat($pattern, new BodyRegExp($this->_response), $verboseMessage);
- }
- /**
- * Asserts that the response body does not match a given regular expression.
- *
- * @param string $pattern The pattern to compare against.
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertResponseNotRegExp(string $pattern, string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat($pattern, new BodyNotRegExp($this->_response), $verboseMessage);
- }
- /**
- * Assert response content is not empty.
- *
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertResponseNotEmpty(string $message = ''): void
- {
- $this->assertThat(null, new BodyNotEmpty($this->_response), $message);
- }
- /**
- * Assert response content is empty.
- *
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertResponseEmpty(string $message = ''): void
- {
- $this->assertThat(null, new BodyEmpty($this->_response), $message);
- }
- /**
- * Asserts that the search string was in the template name.
- *
- * @param string $content The content to check for.
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertTemplate(string $content, string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat($content, new TemplateFileEquals($this->_viewName), $verboseMessage);
- }
- /**
- * Asserts that the search string was in the layout name.
- *
- * @param string $content The content to check for.
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertLayout(string $content, string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat($content, new LayoutFileEquals($this->_layoutName), $verboseMessage);
- }
- /**
- * Asserts session contents
- *
- * @param mixed $expected The expected contents.
- * @param string $path The session data path. Uses Hash::get() compatible notation
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertSession($expected, string $path, string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat($expected, new SessionEquals($this->_requestSession, $path), $verboseMessage);
- }
- /**
- * Asserts a flash message was set
- *
- * @param string $expected Expected message
- * @param string $key Flash key
- * @param string $message Assertion failure message
- * @return void
- */
- public function assertFlashMessage(string $expected, string $key = 'flash', string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat($expected, new FlashParamEquals($this->_requestSession, $key, 'message'), $verboseMessage);
- }
- /**
- * Asserts a flash message was set at a certain index
- *
- * @param int $at Flash index
- * @param string $expected Expected message
- * @param string $key Flash key
- * @param string $message Assertion failure message
- * @return void
- */
- public function assertFlashMessageAt(int $at, string $expected, string $key = 'flash', string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat(
- $expected,
- new FlashParamEquals($this->_requestSession, $key, 'message', $at),
- $verboseMessage
- );
- }
- /**
- * Asserts a flash element was set
- *
- * @param string $expected Expected element name
- * @param string $key Flash key
- * @param string $message Assertion failure message
- * @return void
- */
- public function assertFlashElement(string $expected, string $key = 'flash', string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat(
- $expected,
- new FlashParamEquals($this->_requestSession, $key, 'element'),
- $verboseMessage
- );
- }
- /**
- * Asserts a flash element was set at a certain index
- *
- * @param int $at Flash index
- * @param string $expected Expected element name
- * @param string $key Flash key
- * @param string $message Assertion failure message
- * @return void
- */
- public function assertFlashElementAt(int $at, string $expected, string $key = 'flash', string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat(
- $expected,
- new FlashParamEquals($this->_requestSession, $key, 'element', $at),
- $verboseMessage
- );
- }
- /**
- * Asserts cookie values
- *
- * @param mixed $expected The expected contents.
- * @param string $name The cookie name.
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertCookie($expected, string $name, string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat($name, new CookieSet($this->_response), $verboseMessage);
- $this->assertThat($expected, new CookieEquals($this->_response, $name), $verboseMessage);
- }
- /**
- * Asserts a cookie has not been set in the response
- *
- * @param string $cookie The cookie name to check
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertCookieNotSet(string $cookie, string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat($cookie, new CookieNotSet($this->_response), $verboseMessage);
- }
- /**
- * Disable the error handler middleware.
- *
- * By using this function, exceptions are no longer caught by the ErrorHandlerMiddleware
- * and are instead re-thrown by the TestExceptionRenderer. This can be helpful
- * when trying to diagnose/debug unexpected failures in test cases.
- *
- * @return void
- */
- public function disableErrorHandlerMiddleware(): void
- {
- Configure::write('Error.exceptionRenderer', TestExceptionRenderer::class);
- }
- /**
- * Asserts cookie values which are encrypted by the
- * CookieComponent.
- *
- * The difference from assertCookie() is this decrypts the cookie
- * value like the CookieComponent for this assertion.
- *
- * @param mixed $expected The expected contents.
- * @param string $name The cookie name.
- * @param string $encrypt Encryption mode to use.
- * @param string|null $key Encryption key used. Defaults
- * to Security.salt.
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- * @see \Cake\Utility\CookieCryptTrait::_encrypt()
- */
- public function assertCookieEncrypted(
- $expected,
- string $name,
- string $encrypt = 'aes',
- ?string $key = null,
- string $message = ''
- ): void {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat($name, new CookieSet($this->_response), $verboseMessage);
- $this->_cookieEncryptionKey = $key;
- $this->assertThat(
- $expected,
- new CookieEncryptedEquals($this->_response, $name, $encrypt, $this->_getCookieEncryptionKey())
- );
- }
- /**
- * Asserts that a file with the given name was sent in the response
- *
- * @param string $expected The absolute file path that should be sent in the response.
- * @param string $message The failure message that will be appended to the generated message.
- * @return void
- */
- public function assertFileResponse(string $expected, string $message = ''): void
- {
- $verboseMessage = $this->extractVerboseMessage($message);
- $this->assertThat(null, new FileSent($this->_response), $verboseMessage);
- $this->assertThat($expected, new FileSentAs($this->_response), $verboseMessage);
- }
- /**
- * Inspect controller to extract possible causes of the failed assertion
- *
- * @param string $message Original message to use as a base
- * @return string
- */
- protected function extractVerboseMessage(string $message): string
- {
- if ($this->_exception instanceof Exception) {
- $message .= $this->extractExceptionMessage($this->_exception);
- }
- if ($this->_controller === null) {
- return $message;
- }
- $error = $this->_controller->viewBuilder()->getVar('error');
- if ($error instanceof Exception) {
- $message .= $this->extractExceptionMessage($this->viewVariable('error'));
- }
- return $message;
- }
- /**
- * Extract verbose message for existing exception
- *
- * @param \Exception $exception Exception to extract
- * @return string
- */
- protected function extractExceptionMessage(Exception $exception): string
- {
- return PHP_EOL .
- sprintf('Possibly related to %s: "%s" ', get_class($exception), $exception->getMessage()) .
- PHP_EOL .
- $exception->getTraceAsString();
- }
- }
|