| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813 |
- <?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)
- * @link https://cakephp.org CakePHP(tm) Project
- * @since 3.0.0
- * @license https://opensource.org/licenses/mit-license.php MIT License
- */
- namespace Cake\Test\TestCase\TestSuite;
- use Cake\Controller\Controller;
- use Cake\Core\Configure;
- use Cake\Event\EventManager;
- use Cake\Http\Cookie\Cookie;
- use Cake\Http\Middleware\CsrfProtectionMiddleware;
- use Cake\Http\Middleware\EncryptedCookieMiddleware;
- use Cake\Http\Middleware\SessionCsrfProtectionMiddleware;
- use Cake\Http\Response;
- use Cake\Http\ServerRequest;
- use Cake\Http\Session;
- use Cake\Routing\Route\InflectedRoute;
- use Cake\Routing\RouteBuilder;
- use Cake\Routing\Router;
- use Cake\TestSuite\IntegrationTestTrait;
- use Cake\TestSuite\TestCase;
- use Cake\Utility\Security;
- use Laminas\Diactoros\UploadedFile;
- use LogicException;
- use OutOfBoundsException;
- use PHPUnit\Framework\AssertionFailedError;
- use stdClass;
- use TestApp\ReflectionDependency;
- /**
- * Self test of the IntegrationTestTrait
- */
- class IntegrationTestTraitTest extends TestCase
- {
- use IntegrationTestTrait;
- /**
- * stub encryption key.
- *
- * @var string
- */
- protected $key = 'abcdabcdabcdabcdabcdabcdabcdabcdabcd';
- /**
- * @var \Cake\Routing\RouteBuilder
- */
- protected $builder;
- /**
- * Setup method
- */
- public function setUp(): void
- {
- parent::setUp();
- static::setAppNamespace();
- Router::reload();
- $this->builder = Router::createRouteBuilder('/');
- $this->builder->setExtensions(['json']);
- $this->builder->registerMiddleware('cookie', new EncryptedCookieMiddleware(['secrets'], $this->key));
- $this->builder->applyMiddleware('cookie');
- $this->builder->setRouteClass(InflectedRoute::class);
- $this->builder->get('/get/{controller}/{action}', []);
- $this->builder->head('/head/{controller}/{action}', []);
- $this->builder->options('/options/{controller}/{action}', []);
- $this->builder->connect('/{controller}/{action}/*', []);
- $this->builder->scope('/cookie-csrf/', ['csrf' => 'cookie'], function (RouteBuilder $routes): void {
- $routes->registerMiddleware('cookieCsrf', new CsrfProtectionMiddleware());
- $routes->applyMiddleware('cookieCsrf');
- $routes->connect('/posts/{action}', ['controller' => 'Posts']);
- });
- $this->builder->scope('/session-csrf/', ['csrf' => 'session'], function (RouteBuilder $routes): void {
- $routes->registerMiddleware('sessionCsrf', new SessionCsrfProtectionMiddleware());
- $routes->applyMiddleware('sessionCsrf');
- $routes->connect('/posts/{action}/', ['controller' => 'Posts']);
- });
- $this->configApplication(Configure::read('App.namespace') . '\Application', null);
- }
- /**
- * Tests that all data that used by the request is cast to strings
- */
- public function testDataCastToString(): void
- {
- $data = [
- 'title' => 'Blog Post',
- 'status' => 1,
- 'published' => true,
- 'not_published' => false,
- 'comments' => [
- [
- 'body' => 'Comment',
- 'status' => 1,
- ],
- ],
- 'file' => [
- 'tmp_name' => __FILE__,
- 'size' => 42,
- 'error' => 0,
- 'type' => 'text/plain',
- 'name' => 'Uploaded file',
- ],
- 'pictures' => [
- 'name' => [
- ['file' => 'a-file.png'],
- ['file' => 'a-moose.png'],
- ],
- 'type' => [
- ['file' => 'image/png'],
- ['file' => 'image/jpg'],
- ],
- 'tmp_name' => [
- ['file' => __FILE__],
- ['file' => __FILE__],
- ],
- 'error' => [
- ['file' => 0],
- ['file' => 0],
- ],
- 'size' => [
- ['file' => 17188],
- ['file' => 2010],
- ],
- ],
- 'upload' => new UploadedFile(__FILE__, 42, 0),
- ];
- $request = $this->_buildRequest('/posts/add', 'POST', $data);
- $this->assertIsString($request['post']['status']);
- $this->assertIsString($request['post']['published']);
- $this->assertSame('0', $request['post']['not_published']);
- $this->assertIsString($request['post']['comments'][0]['status']);
- $this->assertIsInt($request['post']['file']['error']);
- $this->assertIsInt($request['post']['file']['size']);
- $this->assertIsInt($request['post']['pictures']['error'][0]['file']);
- $this->assertIsInt($request['post']['pictures']['error'][1]['file']);
- $this->assertIsInt($request['post']['pictures']['size'][0]['file']);
- $this->assertIsInt($request['post']['pictures']['size'][1]['file']);
- $this->assertInstanceOf(UploadedFile::class, $request['post']['upload']);
- }
- /**
- * Test building a request.
- */
- public function testRequestBuilding(): void
- {
- $this->requestAsJson();
- $this->configRequest([
- 'headers' => [
- 'X-CSRF-Token' => 'abc123',
- ],
- 'base' => '',
- 'webroot' => '/',
- 'environment' => [
- 'PHP_AUTH_USER' => 'foo',
- 'PHP_AUTH_PW' => 'bar',
- ],
- ]);
- $this->cookie('split_token', 'def345');
- $this->session(['User' => ['id' => '1', 'username' => 'mark']]);
- $request = $this->_buildRequest('/tasks/add', 'POST', ['title' => 'First post']);
- $this->assertSame('abc123', $request['environment']['HTTP_X_CSRF_TOKEN']);
- $this->assertSame('application/json', $request['environment']['CONTENT_TYPE']);
- $this->assertSame('/tasks/add', $request['url']);
- $this->assertArrayHasKey('split_token', $request['cookies']);
- $this->assertSame('def345', $request['cookies']['split_token']);
- $this->assertSame(['id' => '1', 'username' => 'mark'], $request['session']->read('User'));
- $this->assertSame('foo', $request['environment']['PHP_AUTH_USER']);
- $this->assertSame('bar', $request['environment']['PHP_AUTH_PW']);
- }
- /**
- * Test request building adds csrf tokens
- */
- public function testRequestBuildingCsrfTokens(): void
- {
- $this->enableCsrfToken();
- $request = $this->_buildRequest('/tasks/add', 'POST', ['title' => 'First post']);
- $this->assertArrayHasKey('csrfToken', $request['cookies']);
- $this->assertArrayHasKey('_csrfToken', $request['post']);
- $this->assertSame($request['cookies']['csrfToken'], $request['post']['_csrfToken']);
- $this->assertSame($request['session']->read('csrfToken'), $request['post']['_csrfToken']);
- $this->cookie('csrfToken', '');
- $request = $this->_buildRequest('/tasks/add', 'POST', [
- '_csrfToken' => 'fale',
- 'title' => 'First post',
- ]);
- $this->assertSame('', $request['cookies']['csrfToken']);
- $this->assertSame('fale', $request['post']['_csrfToken']);
- }
- /**
- * Test multiple actions using CSRF tokens don't fail
- */
- public function testEnableCsrfMultipleRequests(): void
- {
- $this->enableCsrfToken();
- $first = $this->_buildRequest('/tasks/add', 'POST', ['title' => 'First post']);
- $second = $this->_buildRequest('/tasks/add', 'POST', ['title' => 'Second post']);
- $this->assertSame(
- $first['cookies']['csrfToken'],
- $second['post']['_csrfToken'],
- 'Csrf token should match cookie'
- );
- $this->assertSame(
- $first['session']->read('csrfToken'),
- $second['post']['_csrfToken'],
- 'Csrf token should match session'
- );
- $this->assertSame(
- $first['post']['_csrfToken'],
- $second['post']['_csrfToken'],
- 'Tokens should be consistent per test method'
- );
- }
- /**
- * Test for issue #17612 - skip adding tokens for GET without data.
- */
- public function testAddTokenInGetRequest(): void
- {
- $this->enableCsrfToken();
- $this->enableSecurityToken();
- $requestWithoutTokens = $this->_buildRequest('tasks/view', 'GET');
- $this->assertArrayNotHasKey('_Token', $requestWithoutTokens['post']);
- $this->assertArrayNotHasKey('_csrfToken', $requestWithoutTokens['post']);
- $this->assertArrayNotHasKey('csrfToken', $requestWithoutTokens['cookies']);
- $this->enableCsrfToken();
- $this->enableSecurityToken();
- $requestWithTokens = $this->_buildRequest('tasks/view', 'GET', ['lorem' => 'ipsum']);
- $this->assertArrayHasKey('_Token', $requestWithTokens['post']);
- $this->assertArrayHasKey('_csrfToken', $requestWithTokens['post']);
- $this->assertArrayHasKey('csrfToken', $requestWithTokens['cookies']);
- }
- /**
- * Test building a request, with query parameters
- */
- public function testRequestBuildingQueryParameters(): void
- {
- $request = $this->_buildRequest('/tasks/view?archived=yes', 'GET', []);
- $this->assertSame('/tasks/view', $request['url']);
- $this->assertSame('archived=yes', $request['environment']['QUERY_STRING']);
- $this->assertSame('/tasks/view', $request['environment']['REQUEST_URI']);
- }
- /**
- * Test cookie encrypted
- *
- * @see CookieComponentControllerTest
- */
- public function testCookieEncrypted(): void
- {
- Security::setSalt($this->key);
- $this->cookieEncrypted('KeyOfCookie', 'Encrypted with aes by default');
- $request = $this->_buildRequest('/tasks/view', 'GET', []);
- $this->assertStringStartsWith('Q2FrZQ==.', $request['cookies']['KeyOfCookie']);
- }
- /**
- * Test sending get request and using default `test_app/config/routes.php`.
- */
- public function testGetUsingApplicationWithPluginRoutes(): void
- {
- // first clean routes to have Router::$initailized === false
- Router::reload();
- $this->clearPlugins();
- $this->configApplication(Configure::read('App.namespace') . '\ApplicationWithPluginRoutes', null);
- $this->get('/test_plugin');
- $this->assertResponseOk();
- }
- /**
- * Test sending get request and using default `test_app/config/routes.php`.
- */
- public function testGetUsingApplicationWithDefaultRoutes(): void
- {
- // first clean routes to have Router::$initialized === false
- Router::reload();
- $this->configApplication(Configure::read('App.namespace') . '\ApplicationWithDefaultRoutes', null);
- $this->get('/some_alias');
- $this->assertResponseOk();
- $this->assertSame('5', $this->_getBodyAsString());
- }
- public function testExceptionsInMiddlewareJsonView(): void
- {
- Router::reload();
- $this->builder->connect('/json_response/api_get_data', [
- 'controller' => 'JsonResponse',
- 'action' => 'apiGetData',
- ]);
- $this->configApplication(Configure::read('App.namespace') . '\ApplicationWithExceptionsInMiddleware', null);
- $this->_request['headers'] = ['Accept' => 'application/json'];
- $this->get('/json_response/api_get_data');
- $this->assertResponseCode(403);
- $this->assertHeader('Content-Type', 'application/json');
- $this->assertResponseContains('"message": "Sample Message"');
- $this->assertResponseContains('"code": 403');
- }
- /**
- * Test sending head requests.
- */
- public function testHead(): void
- {
- $this->assertNull($this->_response);
- $this->head('/request_action/test_request_action');
- $this->assertNotEmpty($this->_response);
- $this->assertInstanceOf('Cake\Http\Response', $this->_response);
- $this->assertResponseSuccess();
- }
- /**
- * Test sending head requests.
- */
- public function testHeadMethodRoute(): void
- {
- $this->head('/head/request_action/test_request_action');
- $this->assertResponseSuccess();
- }
- /**
- * Test sending options requests.
- */
- public function testOptions(): void
- {
- $this->assertNull($this->_response);
- $this->options('/request_action/test_request_action');
- $this->assertNotEmpty($this->_response);
- $this->assertInstanceOf('Cake\Http\Response', $this->_response);
- $this->assertResponseSuccess();
- }
- /**
- * Test sending options requests.
- */
- public function testOptionsMethodRoute(): void
- {
- $this->options('/options/request_action/test_request_action');
- $this->assertResponseSuccess();
- }
- /**
- * Test sending get requests sets the request method
- */
- public function testGetSpecificRouteHttpServer(): void
- {
- $this->get('/get/request_action/test_request_action');
- $this->assertResponseOk();
- $this->assertSame('This is a test', (string)$this->_response->getBody());
- }
- /**
- * Test customizing the app class.
- */
- public function testConfigApplication(): void
- {
- $this->expectException(LogicException::class);
- $this->expectExceptionMessage('Cannot load `TestApp\MissingApp` for use in integration');
- $this->configApplication('TestApp\MissingApp', []);
- $this->get('/request_action/test_request_action');
- }
- /**
- * Test sending get requests with Http\Server
- */
- public function testGetHttpServer(): void
- {
- $this->assertNull($this->_response);
- $this->get('/request_action/test_request_action');
- $this->assertNotEmpty($this->_response);
- $this->assertInstanceOf('Cake\Http\Response', $this->_response);
- $this->assertSame('This is a test', (string)$this->_response->getBody());
- $this->assertHeader('X-Middleware', 'true');
- }
- /**
- * Test that the PSR7 requests get query string data
- */
- public function testGetQueryStringHttpServer(): void
- {
- $this->configRequest(['headers' => ['Content-Type' => 'text/plain']]);
- $this->get('/request_action/params_pass?q=query');
- $this->assertResponseOk();
- $this->assertResponseContains('"q":"query"');
- $this->assertResponseContains('"contentType":"text\/plain"');
- $this->assertHeader('X-Middleware', 'true');
- $request = $this->_controller->getRequest();
- $this->assertStringContainsString('/request_action/params_pass?q=query', $request->getRequestTarget());
- }
- /**
- * Test that the PSR7 requests get query string data
- */
- public function testGetQueryStringSetsHere(): void
- {
- $this->configRequest(['headers' => ['Content-Type' => 'text/plain']]);
- $this->get('/request_action/params_pass?q=query');
- $this->assertResponseOk();
- $this->assertResponseContains('"q":"query"');
- $this->assertResponseContains('"contentType":"text\/plain"');
- $this->assertHeader('X-Middleware', 'true');
- $request = $this->_controller->getRequest();
- $this->assertStringContainsString('/request_action/params_pass?q=query', $request->getRequestTarget());
- $this->assertStringContainsString('/request_action/params_pass', $request->getAttribute('here'));
- }
- /**
- * Test that the PSR7 requests get cookies
- */
- public function testGetCookiesHttpServer(): void
- {
- $this->configRequest(['cookies' => ['split_test' => 'abc']]);
- $this->get('/request_action/cookie_pass');
- $this->assertResponseOk();
- $this->assertResponseContains('"split_test":"abc"');
- $this->assertHeader('X-Middleware', 'true');
- }
- /**
- * Test that the PSR7 requests receive post data
- */
- public function testPostDataHttpServer(): void
- {
- $this->post('/request_action/post_pass', ['title' => 'value']);
- $data = json_decode('' . $this->_response->getBody());
- $this->assertSame('value', $data->title);
- $this->assertHeader('X-Middleware', 'true');
- }
- /**
- * Test that the PSR7 requests receive put data
- */
- public function testPutDataFormUrlEncoded(): void
- {
- $this->configRequest([
- 'headers' => [
- 'Content-Type' => 'application/x-www-form-urlencoded',
- ],
- ]);
- $this->put('/request_action/post_pass', ['title' => 'value']);
- $this->assertResponseOk();
- $data = json_decode('' . $this->_response->getBody());
- $this->assertSame('value', $data->title);
- }
- /**
- * Test that the uploaded files are passed correctly to the request
- */
- public function testUploadedFiles(): void
- {
- $this->configRequest([
- 'files' => [
- 'file' => [
- 'tmp_name' => __FILE__,
- 'size' => 42,
- 'error' => 0,
- 'type' => 'text/plain',
- 'name' => 'Uploaded file',
- ],
- 'pictures' => [
- 'name' => [
- ['file' => 'a-file.png'],
- ['file' => 'a-moose.png'],
- ],
- 'type' => [
- ['file' => 'image/png'],
- ['file' => 'image/jpg'],
- ],
- 'tmp_name' => [
- ['file' => __FILE__],
- ['file' => __FILE__],
- ],
- 'error' => [
- ['file' => 0],
- ['file' => 0],
- ],
- 'size' => [
- ['file' => 17188],
- ['file' => 2010],
- ],
- ],
- 'upload' => new UploadedFile(__FILE__, 42, 0),
- ],
- ]);
- $this->post('/request_action/uploaded_files');
- $this->assertHeader('X-Middleware', 'true');
- $data = json_decode((string)$this->_response->getBody(), true);
- $this->assertSame([
- 'file' => 'Uploaded file',
- 'pictures.0.file' => 'a-file.png',
- 'pictures.1.file' => 'a-moose.png',
- 'upload' => null,
- ], $data);
- }
- /**
- * Test that the PSR7 requests receive encoded data.
- */
- public function testInputDataHttpServer(): void
- {
- $this->post('/request_action/input_test', '{"hello":"world"}');
- if ($this->_response->getBody()->isSeekable()) {
- $this->_response->getBody()->rewind();
- }
- $this->assertSame('world', $this->_response->getBody()->getContents());
- $this->assertHeader('X-Middleware', 'true');
- }
- /**
- * Test that the PSR7 requests receive encoded data.
- */
- public function testInputDataSecurityToken(): void
- {
- $this->enableSecurityToken();
- $this->post('/request_action/input_test', '{"hello":"world"}');
- $this->assertSame('world', '' . $this->_response->getBody());
- $this->assertHeader('X-Middleware', 'true');
- }
- /**
- * Test that the PSR7 requests get cookies
- */
- public function testSessionHttpServer(): void
- {
- $this->session(['foo' => 'session data']);
- $this->get('/request_action/session_test');
- $this->assertResponseOk();
- $this->assertResponseContains('session data');
- $this->assertHeader('X-Middleware', 'true');
- }
- /**
- * Test sending requests stores references to controller/view/layout.
- */
- public function testRequestSetsProperties(): void
- {
- $this->post('/posts/index');
- $this->assertInstanceOf('Cake\Controller\Controller', $this->_controller);
- $this->assertNotEmpty($this->_viewName, 'View name not set');
- $this->assertStringContainsString('templates' . DS . 'Posts' . DS . 'index.php', $this->_viewName);
- $this->assertNotEmpty($this->_layoutName, 'Layout name not set');
- $this->assertStringContainsString('templates' . DS . 'layout' . DS . 'default.php', $this->_layoutName);
- $this->assertTemplate('index');
- $this->assertLayout('default');
- $this->assertSame('value', $this->viewVariable('test'));
- }
- /**
- * Test PSR7 requests store references to controller/view/layout
- */
- public function testRequestSetsPropertiesHttpServer(): void
- {
- $this->post('/posts/index');
- $this->assertInstanceOf('Cake\Controller\Controller', $this->_controller);
- $this->assertNotEmpty($this->_viewName, 'View name not set');
- $this->assertStringContainsString('templates' . DS . 'Posts' . DS . 'index.php', $this->_viewName);
- $this->assertNotEmpty($this->_layoutName, 'Layout name not set');
- $this->assertStringContainsString('templates' . DS . 'layout' . DS . 'default.php', $this->_layoutName);
- $this->assertTemplate('index');
- $this->assertLayout('default');
- $this->assertSame('value', $this->viewVariable('test'));
- }
- /**
- * Tests URLs containing extensions.
- */
- public function testRequestWithExt(): void
- {
- $this->get(['controller' => 'Posts', 'action' => 'ajax', '_ext' => 'json']);
- $this->assertResponseCode(200);
- }
- /**
- * Assert that the stored template doesn't change when cells are rendered.
- */
- public function testAssertTemplateAfterCellRender(): void
- {
- $this->get('/posts/get');
- $this->assertStringContainsString('templates' . DS . 'Posts' . DS . 'get.php', $this->_viewName);
- $this->assertTemplate('get');
- $this->assertResponseContains('cellcontent');
- }
- /**
- * Test array URLs
- */
- public function testArrayUrls(): void
- {
- $this->post(['controller' => 'Posts', 'action' => 'index', '_method' => 'POST']);
- $this->assertResponseOk();
- $this->assertSame('value', $this->viewVariable('test'));
- }
- /**
- * Test array URL with host
- */
- public function testArrayUrlWithHost(): void
- {
- $this->get([
- 'controller' => 'Posts',
- 'action' => 'hostData',
- '_host' => 'app.example.org',
- '_https' => true,
- ]);
- $this->assertResponseOk();
- $this->assertResponseContains('"isSsl":true');
- $this->assertResponseContains('"host":"app.example.org"');
- }
- /**
- * Test array URLs with an empty router.
- */
- public function testArrayUrlsEmptyRouter(): void
- {
- Router::reload();
- $this->assertEmpty(Router::getRouteCollection()->routes());
- $this->get(['controller' => 'Posts', 'action' => 'index']);
- $this->assertResponseOk();
- $this->assertSame('value', $this->viewVariable('test'));
- }
- /**
- * Test flash and cookie assertions
- */
- public function testFlashSessionAndCookieAsserts(): void
- {
- $this->post('/posts/index');
- $this->assertSession('An error message', 'Flash.flash.0.message');
- $this->assertCookie('1', 'remember_me');
- $this->assertCookieNotSet('user_id');
- }
- /**
- * Test flash and cookie assertions
- */
- public function testFlashSessionAndCookieAssertsHttpServer(): void
- {
- $this->post('/posts/index');
- $this->assertSession('An error message', 'Flash.flash.0.message');
- $this->assertCookieNotSet('user_id');
- $this->assertCookie('1', 'remember_me');
- }
- /**
- * Test flash assertions stored with enableRememberFlashMessages() after a
- * redirect.
- */
- public function testFlashAssertionsAfterRedirect(): void
- {
- $this->get('/posts/someRedirect');
- $this->assertResponseCode(302);
- $this->assertSession('A success message', 'Flash.flash.0.message');
- }
- /**
- * Test flash assertions stored with enableRememberFlashMessages() after they
- * are rendered
- */
- public function testFlashAssertionsAfterRender(): void
- {
- $this->enableRetainFlashMessages();
- $this->get('/posts/index/with_flash');
- $this->assertResponseCode(200);
- $this->assertSession('An error message', 'Flash.flash.0.message');
- }
- /**
- * Test flash assertions stored with enableRememberFlashMessages() even if
- * no view is rendered
- */
- public function testFlashAssertionsWithNoRender(): void
- {
- $this->enableRetainFlashMessages();
- $this->get('/posts/flashNoRender');
- $this->assertRedirect();
- $this->assertFlashElement('flash/error');
- $this->assertFlashMessage('An error message');
- }
- /**
- * If multiple requests occur in the same test method
- * flash messages should be retained.
- */
- public function testFlashAssertionMultipleRequests(): void
- {
- $this->enableRetainFlashMessages();
- $this->disableErrorHandlerMiddleware();
- $this->get('/posts/index/with_flash');
- $this->assertResponseCode(200);
- $this->assertFlashMessage('An error message');
- $this->get('/posts/someRedirect');
- $this->assertResponseCode(302);
- $this->assertFlashMessage('A success message');
- }
- /**
- * Test flash assertions stored with enableRememberFlashMessages() even if
- * the controller clears flash data in `beforeRender`
- */
- public function testFlashAssertionsRemoveInBeforeRender(): void
- {
- $this->enableRetainFlashMessages();
- $this->get('/posts/index/with_flash/?clear=true');
- $this->assertResponseOk();
- $this->assertFlashElement('flash/error');
- $this->assertFlashMessage('An error message');
- }
- /**
- * Tests assertCookieNotSet assertion
- */
- public function testAssertCookieNotSet(): void
- {
- $this->cookie('test', 'value');
- $this->get('/posts/index');
- $this->assertCookieNotSet('test');
- $this->get('/posts/redirectWithCookie');
- $this->assertCookieNotSet('test');
- }
- /**
- * Tests the failure message for assertCookieNotSet
- */
- public function testCookieNotSetFailure(): void
- {
- $this->expectException(AssertionFailedError::class);
- $this->expectExceptionMessage('Failed asserting that \'remember_me\' cookie is not set');
- $this->post('/posts/index');
- $this->assertCookieNotSet('remember_me');
- }
- /**
- * Tests the failure message for assertCookieNotSet when no
- * response whas generated
- */
- public function testCookieNotSetFailureNoResponse(): void
- {
- $this->expectException(AssertionFailedError::class);
- $this->expectExceptionMessage('No response set, cannot assert content.');
- $this->assertCookieNotSet('remember_me');
- }
- /**
- * Tests assertCookieIsSet assertion
- */
- public function testAssertCookieIsSet(): void
- {
- $this->get('/posts/secretCookie');
- $this->assertCookieIsSet('secrets');
- $this->get('/posts/redirectWithCookie');
- $this->assertCookieIsSet('remember');
- }
- /**
- * Tests the failure message for assertCookieIsSet
- */
- public function testCookieIsSetFailure(): void
- {
- $this->expectException(AssertionFailedError::class);
- $this->expectExceptionMessage('Failed asserting that \'not-secrets\' cookie is set');
- $this->post('/posts/secretCookie');
- $this->assertCookieIsSet('not-secrets');
- }
- /**
- * Tests the failure message for assertCookieIsSet when no
- * response whas generated
- */
- public function testCookieIsSetFailureNoResponse(): void
- {
- $this->expectException(AssertionFailedError::class);
- $this->expectExceptionMessage('No response set, cannot assert content.');
- $this->assertCookieIsSet('secrets');
- }
- /**
- * Test error handling and error page rendering.
- */
- public function testPostAndErrorHandling(): void
- {
- $this->post('/request_action/error_method');
- $this->assertResponseNotEmpty();
- $this->assertResponseContains('Not there or here');
- $this->assertResponseContains('<!DOCTYPE html>');
- }
- /**
- * Test posting to a secured form action.
- */
- public function testPostSecuredForm(): void
- {
- $this->enableSecurityToken();
- $data = [
- 'title' => 'Some title',
- 'body' => 'Some text',
- ];
- $this->post('/posts/securePost', $data);
- $this->assertResponseOk();
- $this->assertResponseContains('Request was accepted');
- }
- /**
- * Test posting to a secured form action.
- */
- public function testPostSecuredFormNumericField(): void
- {
- $this->enableSecurityToken();
- $data = [
- '123456789' => 'Some text',
- ];
- $this->post('/posts/securePost', $data);
- $this->assertResponseOk();
- $this->assertResponseContains('Request was accepted');
- }
- /**
- * Test posting to a secured form action with nested data.
- */
- public function testPostSecuredFormNestedData(): void
- {
- $this->enableSecurityToken();
- $data = [
- 'title' => 'New post',
- 'comments' => [
- ['comment' => 'A new comment'],
- ],
- 'tags' => ['_ids' => [1, 2, 3, 4]],
- ];
- $this->post('/posts/securePost', $data);
- $this->assertResponseOk();
- $this->assertResponseContains('Request was accepted');
- }
- /**
- * Test posting to a secured form action with unlocked fields
- */
- public function testPostSecuredFormUnlockedFieldsFails(): void
- {
- $this->enableSecurityToken();
- $data = [
- 'title' => 'New post',
- 'comments' => [
- ['comment' => 'A new comment'],
- ],
- 'tags' => ['_ids' => [1, 2, 3, 4]],
- 'some_unlocked_field' => 'Unlocked data',
- ];
- $this->post('/posts/securePost', $data);
- $this->assertResponseCode(400);
- $this->assertResponseContains('Invalid form protection debug token.');
- }
- /**
- * Test posting to a secured form action with unlocked fields
- */
- public function testPostSecuredFormUnlockedFieldsWithSet(): void
- {
- $this->enableSecurityToken();
- $data = [
- 'title' => 'New post',
- 'comments' => [
- ['comment' => 'A new comment'],
- ],
- 'tags' => ['_ids' => [1, 2, 3, 4]],
- 'some_unlocked_field' => 'Unlocked data',
- ];
- $this->setUnlockedFields(['some_unlocked_field']);
- $this->post('/posts/securePost', $data);
- $this->assertResponseOk();
- $this->assertResponseContains('Request was accepted');
- }
- /**
- * Test posting to a secured form action.
- */
- public function testPostSecuredFormWithQuery(): void
- {
- $this->enableSecurityToken();
- $data = [
- 'title' => 'Some title',
- 'body' => 'Some text',
- ];
- $this->post('/posts/securePost?foo=bar', $data);
- $this->assertResponseOk();
- $this->assertResponseContains('Request was accepted');
- }
- /**
- * Test posting to a secured form action with a query that has a part that
- * will be encoded by the security component
- */
- public function testPostSecuredFormWithUnencodedQuery(): void
- {
- $this->enableSecurityToken();
- $data = [
- 'title' => 'Some title',
- 'body' => 'Some text',
- ];
- $this->post('/posts/securePost?foo=/', $data);
- $this->assertResponseOk();
- $this->assertResponseContains('Request was accepted');
- }
- /**
- * Test posting to a secured form action action.
- */
- public function testPostSecuredFormFailure(): void
- {
- $data = [
- 'title' => 'Some title',
- 'body' => 'Some text',
- ];
- $this->post('/posts/securePost', $data);
- $this->assertResponseError();
- }
- /**
- * Integration test for cookie based CSRF token protection success
- */
- public function testPostCookieCsrfSuccess(): void
- {
- $this->enableCsrfToken();
- $data = [
- 'title' => 'Some title',
- 'body' => 'Some text',
- ];
- $this->post('/cookie-csrf/posts/header', $data);
- $this->assertResponseSuccess();
- }
- /**
- * Integration test for cookie based CSRF token protection fail
- */
- public function testPostCookieCsrfFailure(): void
- {
- $this->enableCsrfToken();
- $data = [
- 'title' => 'Some title',
- 'body' => 'Some text',
- '_csrfToken' => 'failure',
- ];
- $this->post('/cookie-csrf/posts/header', $data);
- $this->assertResponseCode(403);
- }
- /**
- * Integration test for session based CSRF token protection success
- */
- public function testPostSessionCsrfSuccess(): void
- {
- $this->enableCsrfToken();
- $data = [
- 'title' => 'Some title',
- 'body' => 'Some text',
- ];
- $this->post('/session-csrf/posts/header', $data);
- $this->assertResponseSuccess();
- }
- /**
- * Integration test for session based CSRF token protection fail
- */
- public function testPostSessionCsrfFailure(): void
- {
- $this->enableCsrfToken();
- $data = [
- 'title' => 'Some title',
- 'body' => 'Some text',
- '_csrfToken' => 'failure',
- ];
- $this->post('/session-csrf/posts/header', $data);
- $this->assertResponseCode(403);
- }
- /**
- * Integration test for session based CSRF token protection success with specified cookie name
- */
- public function testPostSessionCsrfSuccessWithSetCookieName(): void
- {
- $this->builder->scope('/custom-cookie-csrf/', ['csrf' => 'cookie'], function (RouteBuilder $routes): void {
- $routes->registerMiddleware('cookieCsrf', new CsrfProtectionMiddleware(
- [
- 'cookieName' => 'customCsrfToken',
- ]
- ));
- $routes->applyMiddleware('cookieCsrf');
- $routes->connect('/posts/{action}', ['controller' => 'Posts']);
- });
- $this->enableCsrfToken('customCsrfToken');
- $data = [
- 'title' => 'Some title',
- 'body' => 'Some text',
- ];
- $this->post('/custom-cookie-csrf/posts/header', $data);
- $this->assertResponseSuccess();
- }
- /**
- * Integration test for session based CSRF token protection fail with specified cookie name
- */
- public function testPostSessionCsrfFailureWithSetCookieName(): void
- {
- $this->builder->scope('/custom-cookie-csrf/', ['csrf' => 'cookie'], function (RouteBuilder $routes): void {
- $routes->registerMiddleware('cookieCsrf', new CsrfProtectionMiddleware(
- [
- 'cookieName' => 'customCsrfToken',
- ]
- ));
- $routes->applyMiddleware('cookieCsrf');
- $routes->connect('/posts/{action}', ['controller' => 'Posts']);
- });
- $this->enableCsrfToken('customCsrfToken');
- $data = [
- 'title' => 'Some title',
- 'body' => 'Some text',
- '_csrfToken' => 'failure',
- ];
- $this->post('/custom-cookie-csrf/posts/header', $data);
- $this->assertResponseCode(403);
- }
- /**
- * Test that exceptions being thrown are handled correctly.
- */
- public function testWithExpectedException(): void
- {
- $this->get('/tests_apps/throw_exception');
- $this->assertResponseCode(500);
- }
- /**
- * Test that exceptions being thrown are handled correctly by the psr7 stack.
- */
- public function testWithExpectedExceptionHttpServer(): void
- {
- $this->get('/tests_apps/throw_exception');
- $this->assertResponseCode(500);
- }
- /**
- * Test that exceptions being thrown are handled correctly.
- */
- public function testWithUnexpectedException(): void
- {
- $this->expectException(AssertionFailedError::class);
- $this->get('/tests_apps/throw_exception');
- $this->assertResponseCode(501);
- }
- /**
- * Test redirecting and integration tests.
- */
- public function testRedirect(): void
- {
- $this->post('/tests_apps/redirect_to');
- $this->assertResponseSuccess();
- $this->assertResponseCode(302);
- }
- /**
- * Test redirecting and psr7 stack
- */
- public function testRedirectHttpServer(): void
- {
- $this->post('/tests_apps/redirect_to');
- $this->assertResponseCode(302);
- $this->assertHeader('X-Middleware', 'true');
- }
- /**
- * Test redirecting and integration tests.
- */
- public function testRedirectPermanent(): void
- {
- $this->post('/tests_apps/redirect_to_permanent');
- $this->assertResponseSuccess();
- $this->assertResponseCode(301);
- }
- /**
- * Test the responseOk status assertion
- */
- public function testAssertResponseStatusCodes(): void
- {
- $this->_response = new Response();
- $this->_response = $this->_response->withStatus(200);
- $this->assertResponseOk();
- $this->_response = $this->_response->withStatus(201);
- $this->assertResponseOk();
- $this->_response = $this->_response->withStatus(204);
- $this->assertResponseOk();
- $this->_response = $this->_response->withStatus(202);
- $this->assertResponseSuccess();
- $this->_response = $this->_response->withStatus(302);
- $this->assertResponseSuccess();
- $this->_response = $this->_response->withStatus(400);
- $this->assertResponseError();
- $this->_response = $this->_response->withStatus(417);
- $this->assertResponseError();
- $this->_response = $this->_response->withStatus(500);
- $this->assertResponseFailure();
- $this->_response = $this->_response->withStatus(505);
- $this->assertResponseFailure();
- $this->_response = $this->_response->withStatus(301);
- $this->assertResponseCode(301);
- }
- /**
- * Test the location header assertion.
- */
- public function testAssertRedirect(): void
- {
- $this->_response = new Response();
- $this->_response = $this->_response->withHeader('Location', 'http://localhost/get/tasks/index');
- $this->assertRedirect();
- $this->assertRedirect('/get/tasks/index');
- $this->assertRedirect(['controller' => 'Tasks', 'action' => 'index']);
- $this->assertResponseEmpty();
- }
- /**
- * Test the location header assertion.
- */
- public function testAssertRedirectEquals(): void
- {
- $this->_response = new Response();
- $this->_response = $this->_response->withHeader('Location', '/get/tasks/index');
- $this->assertRedirect();
- $this->assertRedirectEquals('/get/tasks/index');
- $this->assertRedirectEquals(['controller' => 'Tasks', 'action' => 'index']);
- $this->assertResponseEmpty();
- }
- /**
- * Test the location header assertion string not contains
- */
- public function testAssertRedirectNotContains(): void
- {
- $this->_response = new Response();
- $this->_response = $this->_response->withHeader('Location', 'http://localhost/tasks/index');
- $this->assertRedirectNotContains('test');
- }
- /**
- * Test the location header assertion.
- */
- public function testAssertNoRedirect(): void
- {
- $this->_response = new Response();
- $this->assertNoRedirect();
- }
- /**
- * Test the location header assertion string contains
- */
- public function testAssertRedirectContains(): void
- {
- $this->_response = new Response();
- $this->_response = $this->_response->withHeader('Location', 'http://localhost/tasks/index');
- $this->assertRedirectContains('/tasks/index');
- }
- /**
- * Test the header assertion.
- */
- public function testAssertHeader(): void
- {
- $this->_response = new Response();
- $this->_response = $this->_response->withHeader('Etag', 'abc123');
- $this->assertHeader('Etag', 'abc123');
- }
- /**
- * Test the header contains assertion.
- */
- public function testAssertHeaderContains(): void
- {
- $this->_response = new Response();
- $this->_response = $this->_response->withHeader('Etag', 'abc123');
- $this->assertHeaderContains('Etag', 'abc');
- }
- /**
- * Test the header not contains assertion.
- */
- public function testAssertHeaderNotContains(): void
- {
- $this->_response = new Response();
- $this->_response = $this->_response->withHeader('Etag', 'abc123');
- $this->assertHeaderNotContains('Etag', 'xyz');
- }
- /**
- * Test the content type assertion.
- */
- public function testAssertContentType(): void
- {
- $this->_response = new Response();
- $this->_response = $this->_response->withType('json');
- $this->assertContentType('json');
- $this->assertContentType('application/json');
- }
- /**
- * Test that type() in an action sets the content-type header.
- */
- public function testContentTypeInAction(): void
- {
- $this->get('/tests_apps/set_type');
- $this->assertHeader('Content-Type', 'application/json');
- $this->assertContentType('json');
- $this->assertContentType('application/json');
- }
- /**
- * Test the content assertion.
- */
- public function testAssertResponseEquals(): void
- {
- $this->_response = new Response();
- $this->_response = $this->_response->withStringBody('Some content');
- $this->assertResponseEquals('Some content');
- }
- /**
- * Test the negated content assertion.
- */
- public function testAssertResponseNotEquals(): void
- {
- $this->_response = new Response();
- $this->_response = $this->_response->withStringBody('Some content');
- $this->assertResponseNotEquals('Some Content');
- }
- /**
- * Test the content assertion.
- */
- public function testAssertResponseContains(): void
- {
- $this->_response = new Response();
- $this->_response = $this->_response->withStringBody('Some content');
- $this->assertResponseContains('content');
- }
- /**
- * Test the content assertion with no case sensitivity.
- */
- public function testAssertResponseContainsWithIgnoreCaseFlag(): void
- {
- $this->_response = new Response();
- $this->_response = $this->_response->withStringBody('Some content');
- $this->assertResponseContains('some', 'Failed asserting that the body contains given content', true);
- }
- /**
- * Test the negated content assertion.
- */
- public function testAssertResponseNotContains(): void
- {
- $this->_response = new Response();
- $this->_response = $this->_response->withStringBody('Some content');
- $this->assertResponseNotContains('contents');
- }
- /**
- * Test the content regexp assertion.
- */
- public function testAssertResponseRegExp(): void
- {
- $this->_response = new Response();
- $this->_response = $this->_response->withStringBody('Some content');
- $this->assertResponseRegExp('/cont/');
- }
- /**
- * Test the content regexp assertion failing
- */
- public function testAssertResponseRegExpNoResponse(): void
- {
- $this->expectException(AssertionFailedError::class);
- $this->expectExceptionMessage('No response set');
- $this->assertResponseRegExp('/cont/');
- }
- /**
- * Test the negated content regexp assertion.
- */
- public function testAssertResponseNotRegExp(): void
- {
- $this->_response = new Response();
- $this->_response = $this->_response->withStringBody('Some content');
- $this->assertResponseNotRegExp('/cant/');
- }
- /**
- * Test negated content regexp assertion failing
- */
- public function testAssertResponseNotRegExpNoResponse(): void
- {
- $this->expectException(AssertionFailedError::class);
- $this->expectExceptionMessage('No response set');
- $this->assertResponseNotRegExp('/cont/');
- }
- /**
- * Test that works in tandem with testEventManagerReset2 to
- * test the EventManager reset.
- *
- * The return value is passed to testEventManagerReset2 as
- * an arguments.
- */
- public function testEventManagerReset1(): EventManager
- {
- $eventManager = EventManager::instance();
- $this->assertInstanceOf('Cake\Event\EventManager', $eventManager);
- return $eventManager;
- }
- /**
- * Test if the EventManager is reset between tests.
- *
- * @depends testEventManagerReset1
- */
- public function testEventManagerReset2(EventManager $prevEventManager): void
- {
- $this->assertInstanceOf('Cake\Event\EventManager', $prevEventManager);
- $this->assertNotSame($prevEventManager, EventManager::instance());
- }
- /**
- * Test sending file in requests.
- */
- public function testSendFile(): void
- {
- $this->get('/posts/file');
- $this->assertFileResponse(TEST_APP . 'TestApp' . DS . 'Controller' . DS . 'PostsController.php');
- }
- /**
- * Test sending file in requests.
- */
- public function testSendUnlinked(): void
- {
- $file = microtime(true) . 'txt';
- $path = TMP . $file;
- file_put_contents($path, 'testing unlink');
- $this->get("/posts/file?file={$file}");
- $this->assertResponseOk();
- $this->assertFileResponse($path);
- $this->assertFileExists($path);
- system("rm -rf {$path}");
- $this->assertFileDoesNotExist($path);
- }
- /**
- * Test sending file with psr7 stack
- */
- public function testSendFileHttpServer(): void
- {
- $this->get('/posts/file');
- $this->assertFileResponse(TEST_APP . 'TestApp' . DS . 'Controller' . DS . 'PostsController.php');
- }
- /**
- * Test that assertFile requires a response
- */
- public function testAssertFileNoResponse(): void
- {
- $this->expectException(AssertionFailedError::class);
- $this->expectExceptionMessage('No response set, cannot assert content');
- $this->assertFileResponse('foo');
- }
- /**
- * Test that assertFile requires a file
- */
- public function testAssertFileNoFile(): void
- {
- $this->expectException(AssertionFailedError::class);
- $this->expectExceptionMessage('Failed asserting that file was sent.');
- $this->get('/posts/get');
- $this->assertFileResponse('foo');
- }
- /**
- * Test disabling the error handler middleware with exceptions
- * in controllers.
- */
- public function testDisableErrorHandlerMiddleware(): void
- {
- $this->expectException(OutOfBoundsException::class);
- $this->expectExceptionMessage('oh no!');
- $this->disableErrorHandlerMiddleware();
- $this->get('/posts/throw_exception');
- }
- /**
- * tests getting a secure action while passing a query string
- *
- * @dataProvider methodsProvider
- */
- public function testSecureWithQueryString(string $method): void
- {
- $this->enableSecurityToken();
- $this->{$method}('/posts/securePost/?ids[]=1&ids[]=2');
- $this->assertResponseOk();
- }
- /**
- * Tests flash assertions
- *
- * @throws \PHPUnit\Exception
- */
- public function testAssertFlashMessage(): void
- {
- $this->get('/posts/stacked_flash');
- $this->assertFlashElement('flash/error');
- $this->assertFlashElement('flash/success', 'custom');
- $this->assertFlashMessage('Error 1');
- $this->assertFlashMessageAt(0, 'Error 1');
- $this->assertFlashElementAt(0, 'flash/error');
- $this->assertFlashMessage('Error 2');
- $this->assertFlashMessageAt(1, 'Error 2');
- $this->assertFlashElementAt(1, 'flash/error');
- $this->assertFlashMessage('Success 1', 'custom');
- $this->assertFlashMessageAt(0, 'Success 1', 'custom');
- $this->assertFlashElementAt(0, 'flash/success', 'custom');
- $this->assertFlashMessage('Success 2', 'custom');
- $this->assertFlashMessageAt(1, 'Success 2', 'custom');
- $this->assertFlashElementAt(1, 'flash/success', 'custom');
- }
- /**
- * Tests asserting flash messages without first sending a request
- */
- public function testAssertFlashMessageWithoutSendingRequest(): void
- {
- $this->expectException(AssertionFailedError::class);
- $message = 'There is no stored session data. Perhaps you need to run a request?';
- $message .= ' Additionally, ensure `$this->enableRetainFlashMessages()` has been enabled for the test.';
- $this->expectExceptionMessage($message);
- $this->assertFlashMessage('Will not work');
- }
- /**
- * tests failure messages for assertions
- *
- * @param string $assertion Assertion method
- * @param string $message Expected failure message
- * @param string $url URL to test
- * @param mixed ...$rest
- * @dataProvider assertionFailureMessagesProvider
- */
- public function testAssertionFailureMessages($assertion, $message, $url, ...$rest): void
- {
- $this->expectException(AssertionFailedError::class);
- $this->expectExceptionMessage($message);
- Security::setSalt($this->key);
- $this->get($url);
- call_user_func_array($this->$assertion(...), $rest);
- }
- /**
- * Test for assertion message generation for previous.
- *
- * @return void
- */
- public function testAssertMessagePrevious()
- {
- $this->expectException(AssertionFailedError::class);
- $this->expectExceptionMessage('Caused by `RuntimeException`');
- $this->get('/posts/throw_chained');
- $this->assertContentType('test');
- }
- /**
- * data provider for assertion failure messages
- *
- * @return array
- */
- public static function assertionFailureMessagesProvider(): array
- {
- $templateDir = TEST_APP . 'templates' . DS;
- return [
- 'assertContentType' => ['assertContentType', 'Failed asserting that \'test\' is set as the Content-Type (`text/html`).', '/posts/index', 'test'],
- 'assertContentTypeVerbose' => ['assertContentType', 'Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."', '/notfound', 'test'],
- 'assertCookie' => ['assertCookie', 'Failed asserting that \'test\' is in cookie \'remember_me\'.', '/posts/index', 'test', 'remember_me'],
- 'assertCookieVerbose' => ['assertCookie', 'Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."', '/notfound', 'test', 'remember_me'],
- 'assertCookieEncrypted' => ['assertCookieEncrypted', 'Failed asserting that \'test\' is encrypted in cookie \'secrets\'.', '/posts/secretCookie', 'test', 'secrets'],
- 'assertCookieEncryptedVerbose' => ['assertCookieEncrypted', 'Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."', '/notfound', 'test', 'NameOfCookie'],
- 'assertCookieNotSet' => ['assertCookieNotSet', 'Failed asserting that \'remember_me\' cookie is not set.', '/posts/index', 'remember_me'],
- 'assertFileResponse' => ['assertFileResponse', 'Failed asserting that \'test\' file was sent.', '/posts/file', 'test'],
- 'assertFileResponseVerbose' => ['assertFileResponse', 'Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."', '/notfound', 'test'],
- 'assertHeader' => ['assertHeader', 'Failed asserting that \'test\' equals content in header \'X-Cake\' (`custom header`).', '/posts/header', 'X-Cake', 'test'],
- 'assertHeaderContains' => ['assertHeaderContains', 'Failed asserting that \'test\' is in header \'X-Cake\' (`custom header`)', '/posts/header', 'X-Cake', 'test'],
- 'assertHeaderNotContains' => ['assertHeaderNotContains', 'Failed asserting that \'custom header\' is not in header \'X-Cake\' (`custom header`)', '/posts/header', 'X-Cake', 'custom header'],
- 'assertHeaderContainsVerbose' => ['assertHeaderContains', 'Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."', '/notfound', 'X-Cake', 'test'],
- 'assertHeaderNotContainsVerbose' => ['assertHeaderNotContains', 'Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."', '/notfound', 'X-Cake', 'test'],
- 'assertLayout' => ['assertLayout', 'Failed asserting that \'custom_layout\' equals layout file `' . $templateDir . 'layout' . DS . 'default.php`.', '/posts/index', 'custom_layout'],
- 'assertLayoutVerbose' => ['assertLayout', 'Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."', '/notfound', 'custom_layout'],
- 'assertRedirect' => ['assertRedirect', 'Failed asserting that \'http://localhost/\' equals content in header \'Location\' (`http://localhost/posts`).', '/posts/flashNoRender', '/'],
- 'assertRedirectVerbose' => ['assertRedirect', 'Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."', '/notfound', '/'],
- 'assertRedirectContains' => ['assertRedirectContains', 'Failed asserting that \'/posts/somewhere-else\' is in header \'Location\' (`http://localhost/posts`).', '/posts/flashNoRender', '/posts/somewhere-else'],
- 'assertRedirectContainsVerbose' => ['assertRedirectContains', 'Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."', '/notfound', '/posts/somewhere-else'],
- 'assertRedirectNotContainsVerbose' => ['assertRedirectNotContains', 'Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."', '/notfound', '/posts/somewhere-else'],
- 'assertResponseCode' => ['assertResponseCode', 'Failed asserting that `302` matches response status code `200`.', '/posts/index', 302],
- 'assertResponseContains' => ['assertResponseContains', 'Failed asserting that \'test\' is in response body.', '/posts/index', 'test'],
- 'assertResponseEmpty' => ['assertResponseEmpty', 'Failed asserting that response body is empty.', '/posts/index'],
- 'assertResponseEquals' => ['assertResponseEquals', 'Failed asserting that \'test\' matches response body.', '/posts/index', 'test'],
- 'assertResponseEqualsVerbose' => ['assertResponseEquals', 'Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."', '/notfound', 'test'],
- 'assertResponseError' => ['assertResponseError', 'Failed asserting that 200 is between 400 and 429.', '/posts/index'],
- 'assertResponseFailure' => ['assertResponseFailure', 'Failed asserting that 200 is between 500 and 505.', '/posts/index'],
- 'assertResponseNotContains' => ['assertResponseNotContains', 'Failed asserting that \'index\' is not in response body.', '/posts/index', 'index'],
- 'assertResponseNotEmpty' => ['assertResponseNotEmpty', 'Failed asserting that response body is not empty.', '/posts/empty_response'],
- 'assertResponseNotEquals' => ['assertResponseNotEquals', 'Failed asserting that \'posts index\' does not match response body.', '/posts/index/error', 'posts index'],
- 'assertResponseNotRegExp' => ['assertResponseNotRegExp', 'Failed asserting that `/index/` PCRE pattern not found in response body.', '/posts/index/error', '/index/'],
- 'assertResponseNotRegExpVerbose' => ['assertResponseNotRegExp', 'Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."', '/notfound', '/index/'],
- 'assertResponseOk' => ['assertResponseOk', 'Failed asserting that 404 is between 200 and 204.', '/posts/missing', '/index/'],
- 'assertResponseRegExp' => ['assertResponseRegExp', 'Failed asserting that `/test/` PCRE pattern found in response body.', '/posts/index/error', '/test/'],
- 'assertResponseSuccess' => ['assertResponseSuccess', 'Failed asserting that 404 is between 200 and 308.', '/posts/missing'],
- 'assertResponseSuccessVerbose' => ['assertResponseSuccess', 'Possibly related to `Cake\Controller\Exception\MissingActionException`: "Action `PostsController::missing()` could not be found, or is not accessible."', '/posts/missing'],
- 'assertSession' => ['assertSession', 'Failed asserting that \'test\' is in session path \'Missing.path\'.', '/posts/index', 'test', 'Missing.path'],
- 'assertSessionHasKey' => ['assertSessionHasKey', 'Failed asserting that \'Missing.path\' is a path present in the session.', '/posts/index', 'Missing.path'],
- 'assertSessionNotHasKey' => ['assertSessionNotHasKey', 'Failed asserting that \'Flash.flash\' is not a path present in the session.', '/posts/index', 'Flash.flash'],
- 'assertTemplate' => ['assertTemplate', 'Failed asserting that \'custom_template\' equals template file `' . $templateDir . 'Posts' . DS . 'index.php`.', '/posts/index', 'custom_template'],
- 'assertTemplateVerbose' => ['assertTemplate', 'Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."', '/notfound', 'custom_template'],
- 'assertFlashMessage' => ['assertFlashMessage', 'Failed asserting that \'missing\' is in \'flash\' message.', '/posts/index', 'missing'],
- 'assertFlashMessageWithKey' => ['assertFlashMessage', 'Failed asserting that \'missing\' is in \'auth\' message.', '/posts/index', 'missing', 'auth'],
- 'assertFlashMessageAt' => ['assertFlashMessageAt', 'Failed asserting that \'missing\' is in \'flash\' message #0.', '/posts/index', 0, 'missing'],
- 'assertFlashMessageAtWithKey' => ['assertFlashMessageAt', 'Failed asserting that \'missing\' is in \'auth\' message #0.', '/posts/index', 0, 'missing', 'auth'],
- 'assertFlashElement' => ['assertFlashElement', 'Failed asserting that \'missing\' is in \'flash\' element.', '/posts/index', 'missing'],
- 'assertFlashElementWithKey' => ['assertFlashElement', 'Failed asserting that \'missing\' is in \'auth\' element.', '/posts/index', 'missing', 'auth'],
- 'assertFlashElementAt' => ['assertFlashElementAt', 'Failed asserting that \'missing\' is in \'flash\' element #0.', '/posts/index', 0, 'missing'],
- 'assertFlashElementAtWithKey' => ['assertFlashElementAt', 'Failed asserting that \'missing\' is in \'auth\' element #0.', '/posts/index', 0, 'missing', 'auth'],
- ];
- }
- /**
- * data provider for HTTP methods
- *
- * @return array
- */
- public static function methodsProvider(): array
- {
- return [
- 'GET' => ['get'],
- 'POST' => ['post'],
- 'PATCH' => ['patch'],
- 'PUT' => ['put'],
- 'DELETE' => ['delete'],
- ];
- }
- /**
- * Test assertCookieNotSet is creating a verbose message
- */
- public function testAssertCookieNotSetVerbose(): void
- {
- $this->expectException(AssertionFailedError::class);
- $this->expectExceptionMessage('Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."');
- $this->get('/notfound');
- $this->_response = $this->_response->withCookie(new Cookie('cookie', 'value'));
- $this->assertCookieNotSet('cookie');
- }
- /**
- * Test assertNoRedirect is creating a verbose message
- */
- public function testAssertNoRedirectVerbose(): void
- {
- $this->expectException(AssertionFailedError::class);
- $this->expectExceptionMessage('Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."');
- $this->get('/notfound');
- $this->_response = $this->_response->withHeader('Location', '/redirect');
- $this->assertNoRedirect();
- }
- /**
- * Test the header assertion generating a verbose message.
- */
- public function testAssertHeaderVerbose(): void
- {
- $this->expectException(AssertionFailedError::class);
- $this->expectExceptionMessage('Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."');
- $this->get('/notfound');
- $this->assertHeader('Etag', 'abc123');
- }
- /**
- * Test the assertResponseNotEquals generates a verbose message.
- */
- public function testAssertResponseNotEqualsVerbose(): void
- {
- $this->expectException(AssertionFailedError::class);
- $this->expectExceptionMessage('Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."');
- $this->get('/notfound');
- $this->_response = $this->_response->withStringBody('body');
- $this->assertResponseNotEquals('body');
- }
- /**
- * Test the assertResponseRegExp generates a verbose message.
- */
- public function testAssertResponseRegExpVerbose(): void
- {
- $this->expectException(AssertionFailedError::class);
- $this->expectExceptionMessage('Possibly related to `Cake\Routing\Exception\MissingRouteException`: "A route matching `/notfound` could not be found."');
- $this->get('/notfound');
- $this->_response = $this->_response->withStringBody('body');
- $this->assertResponseRegExp('/patternNotFound/');
- }
- /**
- * Test the assertion generates a verbose message for session related checks.
- *
- * @dataProvider assertionFailureSessionVerboseProvider
- * @param mixed ...$rest
- */
- public function testAssertSessionRelatedVerboseMessages(string $assertMethod, ...$rest): void
- {
- $this->expectException(AssertionFailedError::class);
- $this->expectExceptionMessage('Possibly related to `OutOfBoundsException`: "oh no!"');
- $this->get('/posts/throw_exception');
- $this->_requestSession = new Session();
- call_user_func_array($this->$assertMethod(...), $rest);
- }
- /**
- * data provider for assertion verbose session related tests
- *
- * @return array
- */
- public static function assertionFailureSessionVerboseProvider(): array
- {
- return [
- 'assertFlashMessageVerbose' => ['assertFlashMessage', 'notfound'],
- 'assertFlashMessageAtVerbose' => ['assertFlashMessageAt', 2, 'notfound'],
- 'assertFlashElementVerbose' => ['assertFlashElement', 'notfound'],
- 'assertSessionVerbose' => ['assertSession', 'notfound', 'notfound'],
- ];
- }
- /**
- * Test viewVariable not found
- */
- public function testViewVariableNotFoundShouldReturnNull(): void
- {
- $this->_controller = new Controller(new ServerRequest());
- $this->assertNull($this->viewVariable('notFound'));
- }
- /**
- * Integration test for a controller with action dependencies.
- */
- public function testHandleWithContainerDependencies(): void
- {
- $this->get('/dependencies/requiredDep');
- $this->assertResponseOk();
- $this->assertResponseContains('"key":"value"', 'Contains the data from the stdClass container object.');
- }
- /**
- * Test that mockService() injects into controllers.
- */
- public function testHandleWithMockServices(): void
- {
- $this->mockService(stdClass::class, function () {
- return json_decode('{"mock":true}');
- });
- $this->get('/dependencies/requiredDep');
- $this->assertResponseOk();
- $this->assertResponseContains('"mock":true', 'Contains the data from the stdClass mock container.');
- }
- /**
- * Test that mockService() injects into controllers.
- */
- public function testHandleWithMockServicesFromReflectionContainer(): void
- {
- $this->mockService(ReflectionDependency::class, function () {
- return new ReflectionDependency();
- });
- $this->get('/dependencies/reflectionDep');
- $this->assertResponseOk();
- $this->assertResponseContains('{"dep":{}}', 'Contains the data from the reflection container');
- }
- /**
- * Test that mockService() injects into controllers.
- */
- public function testHandleWithMockServicesOverwrite(): void
- {
- $this->mockService(stdClass::class, function () {
- return json_decode('{"first":true}');
- });
- $this->mockService(stdClass::class, function () {
- return json_decode('{"second":true}');
- });
- $this->get('/dependencies/requiredDep');
- $this->assertResponseOk();
- $this->assertResponseContains('"second":true', 'Contains the data from the stdClass mock container.');
- }
- /**
- * Test that removeMock() unsets mocks
- */
- public function testHandleWithMockServicesUnset(): void
- {
- $this->mockService(stdClass::class, function () {
- return json_decode('{"first":true}');
- });
- $this->removeMockService(stdClass::class);
- $this->get('/dependencies/requiredDep');
- $this->assertResponseOk();
- $this->assertResponseNotContains('"first":true');
- }
- }
|