You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
336 lines
8.0 KiB
336 lines
8.0 KiB
<?php
|
|
/**
|
|
* CodeIgniter
|
|
*
|
|
* An open source application development framework for PHP
|
|
*
|
|
* This content is released under the MIT License (MIT)
|
|
*
|
|
* Copyright (c) 2014-2019 British Columbia Institute of Technology
|
|
* Copyright (c) 2019 CodeIgniter Foundation
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
* in the Software without restriction, including without limitation the rights
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
* THE SOFTWARE.
|
|
*
|
|
* @package CodeIgniter
|
|
* @author CodeIgniter Dev Team
|
|
* @copyright 2019 CodeIgniter Foundation
|
|
* @license https://opensource.org/licenses/MIT MIT License
|
|
* @link https://codeigniter.com
|
|
* @since Version 4.0.0
|
|
* @filesource
|
|
*/
|
|
|
|
namespace CodeIgniter\Test;
|
|
|
|
use CodeIgniter\HTTP\URI;
|
|
use CodeIgniter\HTTP\Request;
|
|
use CodeIgniter\Events\Events;
|
|
use CodeIgniter\HTTP\UserAgent;
|
|
use CodeIgniter\HTTP\IncomingRequest;
|
|
use Config\App;
|
|
use Config\Services;
|
|
|
|
/**
|
|
* Class FeatureTestCase
|
|
*
|
|
* Provides additional utilities for doing full HTTP testing
|
|
* against your application.
|
|
*
|
|
* @package CodeIgniter\Test
|
|
*/
|
|
class FeatureTestCase extends CIDatabaseTestCase
|
|
{
|
|
|
|
/**
|
|
* If present, will override application
|
|
* routes when using call().
|
|
*
|
|
* @var \CodeIgniter\Router\RouteCollection
|
|
*/
|
|
protected $routes;
|
|
|
|
/**
|
|
* Values to be set in the SESSION global
|
|
* before running the test.
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $session = [];
|
|
|
|
/**
|
|
* Enabled auto clean op buffer after request call
|
|
*
|
|
* @var boolean
|
|
*/
|
|
protected $clean = true;
|
|
|
|
/**
|
|
* Sets a RouteCollection that will override
|
|
* the application's route collection.
|
|
*
|
|
* Example routes:
|
|
* [
|
|
* ['get', 'home', 'Home::index']
|
|
* ]
|
|
*
|
|
* @param array $routes
|
|
*
|
|
* @return $this
|
|
*/
|
|
protected function withRoutes(array $routes = null)
|
|
{
|
|
$collection = Services::routes();
|
|
|
|
if ($routes)
|
|
{
|
|
$collection->resetRoutes();
|
|
foreach ($routes as $route)
|
|
{
|
|
$collection->{$route[0]}($route[1], $route[2]);
|
|
}
|
|
}
|
|
|
|
$this->routes = $collection;
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Sets any values that should exist during this session.
|
|
*
|
|
* @param array $values
|
|
*
|
|
* @return $this
|
|
*/
|
|
public function withSession(array $values)
|
|
{
|
|
$this->session = $values;
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Don't run any events while running this test.
|
|
*
|
|
* @return $this
|
|
*/
|
|
public function skipEvents()
|
|
{
|
|
Events::simulate(true);
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Calls a single URI, executes it, and returns a FeatureResponse
|
|
* instance that can be used to run many assertions against.
|
|
*
|
|
* @param string $method
|
|
* @param string $path
|
|
* @param array|null $params
|
|
*
|
|
* @return \CodeIgniter\Test\FeatureResponse
|
|
* @throws \CodeIgniter\Router\Exceptions\RedirectException
|
|
* @throws \Exception
|
|
*/
|
|
public function call(string $method, string $path, array $params = null)
|
|
{
|
|
// Simulate having a blank session
|
|
$_SESSION = [];
|
|
$_SERVER['REQUEST_METHOD'] = $method;
|
|
|
|
$request = $this->setupRequest($method, $path, $params);
|
|
$request = $this->populateGlobals($method, $request, $params);
|
|
|
|
// Make sure the RouteCollection knows what method we're using...
|
|
if (! empty($this->routes))
|
|
{
|
|
$this->routes->setHTTPVerb($method);
|
|
}
|
|
|
|
// Make sure any other classes that might call the request
|
|
// instance get the right one.
|
|
Services::injectMock('request', $request);
|
|
|
|
$response = $this->app
|
|
->setRequest($request)
|
|
->run($this->routes, true);
|
|
|
|
$output = ob_get_contents();
|
|
if (empty($response->getBody()) && ! empty($output))
|
|
{
|
|
$response->setBody($output);
|
|
}
|
|
|
|
// Clean up any open output buffers
|
|
// not relevant to unit testing
|
|
// @codeCoverageIgnoreStart
|
|
|
|
if (ob_get_level() > 0 && $this->clean)
|
|
{
|
|
ob_end_clean();
|
|
}
|
|
// @codeCoverageIgnoreEnd
|
|
|
|
$featureResponse = new FeatureResponse($response);
|
|
|
|
return $featureResponse;
|
|
}
|
|
|
|
/**
|
|
* Performs a GET request.
|
|
*
|
|
* @param string $path
|
|
* @param array|null $params
|
|
*
|
|
* @return \CodeIgniter\Test\FeatureResponse
|
|
* @throws \CodeIgniter\Router\Exceptions\RedirectException
|
|
* @throws \Exception
|
|
*/
|
|
public function get(string $path, array $params = null)
|
|
{
|
|
return $this->call('get', $path, $params);
|
|
}
|
|
|
|
/**
|
|
* Performs a POST request.
|
|
*
|
|
* @param string $path
|
|
* @param array|null $params
|
|
*
|
|
* @return \CodeIgniter\Test\FeatureResponse
|
|
* @throws \CodeIgniter\Router\Exceptions\RedirectException
|
|
* @throws \Exception
|
|
*/
|
|
public function post(string $path, array $params = null)
|
|
{
|
|
return $this->call('post', $path, $params);
|
|
}
|
|
|
|
/**
|
|
* Performs a PUT request
|
|
*
|
|
* @param string $path
|
|
* @param array|null $params
|
|
*
|
|
* @return \CodeIgniter\Test\FeatureResponse
|
|
* @throws \CodeIgniter\Router\Exceptions\RedirectException
|
|
* @throws \Exception
|
|
*/
|
|
public function put(string $path, array $params = null)
|
|
{
|
|
return $this->call('put', $path, $params);
|
|
}
|
|
|
|
/**
|
|
* Performss a PATCH request
|
|
*
|
|
* @param string $path
|
|
* @param array|null $params
|
|
*
|
|
* @return \CodeIgniter\Test\FeatureResponse
|
|
* @throws \CodeIgniter\Router\Exceptions\RedirectException
|
|
* @throws \Exception
|
|
*/
|
|
public function patch(string $path, array $params = null)
|
|
{
|
|
return $this->call('patch', $path, $params);
|
|
}
|
|
|
|
/**
|
|
* Performs a DELETE request.
|
|
*
|
|
* @param string $path
|
|
* @param array|null $params
|
|
*
|
|
* @return \CodeIgniter\Test\FeatureResponse
|
|
* @throws \CodeIgniter\Router\Exceptions\RedirectException
|
|
* @throws \Exception
|
|
*/
|
|
public function delete(string $path, array $params = null)
|
|
{
|
|
return $this->call('delete', $path, $params);
|
|
}
|
|
|
|
/**
|
|
* Performs an OPTIONS request.
|
|
*
|
|
* @param string $path
|
|
* @param array|null $params
|
|
*
|
|
* @return \CodeIgniter\Test\FeatureResponse
|
|
* @throws \CodeIgniter\Router\Exceptions\RedirectException
|
|
* @throws \Exception
|
|
*/
|
|
public function options(string $path, array $params = null)
|
|
{
|
|
return $this->call('options', $path, $params);
|
|
}
|
|
|
|
/**
|
|
* Setup a Request object to use so that CodeIgniter
|
|
* won't try to auto-populate some of the items.
|
|
*
|
|
* @param string $method
|
|
* @param string|null $path
|
|
* @param array|null $params
|
|
*
|
|
* @return \CodeIgniter\HTTP\IncomingRequest
|
|
*/
|
|
protected function setupRequest(string $method, string $path = null, array $params = null): IncomingRequest
|
|
{
|
|
$config = config(App::class);
|
|
$uri = new URI($config->baseURL . '/' . trim($path, '/ '));
|
|
|
|
$request = new IncomingRequest($config, clone($uri), $params, new UserAgent());
|
|
$request->uri = $uri;
|
|
|
|
$request->setMethod($method);
|
|
$request->setProtocolVersion('1.1');
|
|
|
|
return $request;
|
|
}
|
|
|
|
/**
|
|
* Populates the data of our Request with "global" data
|
|
* relevant to the request, like $_POST data.
|
|
*
|
|
* Always populate the GET vars based on the URI.
|
|
*
|
|
* @param string $method
|
|
* @param \CodeIgniter\HTTP\Request $request
|
|
* @param array|null $params
|
|
*
|
|
* @return \CodeIgniter\HTTP\Request
|
|
* @throws \ReflectionException
|
|
*/
|
|
protected function populateGlobals(string $method, Request $request, array $params = null)
|
|
{
|
|
$request->setGlobal('get', $this->getPrivateProperty($request->uri, 'query'));
|
|
if ($method !== 'get')
|
|
{
|
|
$request->setGlobal($method, $params);
|
|
}
|
|
|
|
$_SESSION = $this->session;
|
|
|
|
return $request;
|
|
}
|
|
|
|
}
|
|
|