This class is a transparent base class for Request_Client_HTTP and should not be accessed directly.
Request_Client_External HTTP driver performs external requests using the php-http extension. To use this driver, ensure the following is completed before executing an external request- ideally in the application bootstrap.
Class declared in SYSPATH/classes/Kohana/Request/Client/HTTP.php on line 19.
string $clientdefines the external client to use by default
string(19) "Request_Client_Curl"Cache $_cacheCaching library for request caching
NULL
int $_callback_depthTracks the callback depth of the currently executing request
integer 1
array $_callback_paramsArbitrary parameters that are shared with header callbacks through their Request_Client object
array(0) bool $_followShould redirects be followed?
bool FALSE
array $_follow_headersHeaders to preserve when following a redirect
array(1) ( 0 => string(13) "authorization" )
array $_header_callbacksCallbacks to use when response contains given headers
array(1) ( "Location" => string(34) "Request_Client::on_header_location" )
int $_max_callback_depthMaximum number of requests that header callbacks can trigger before the request is aborted
integer 5
array $_optionscurl options
array(0) bool $_strict_redirectFollow 302 redirect with original request method?
bool TRUE
Creates a new Request_Client object,
allows for dependency injection.
array
$params
 = array(0)  - Paramspublic function __construct(array $params = array())
{
	// Check that PECL HTTP supports requests
	if ( ! http_support(HTTP_SUPPORT_REQUESTS))
	{
		throw new Request_Exception('Need HTTP request support!');
	}
	// Carry on
	parent::__construct($params);
}Sends the HTTP message Request to a remote server and processes the response.
Request
$request
required - Request to sendResponse
$response
required - $request  response to sendResponsepublic function _send_message(Request $request, Response $response)
{
	$http_method_mapping = array(
		HTTP_Request::GET     => HTTPRequest::METH_GET,
		HTTP_Request::HEAD    => HTTPRequest::METH_HEAD,
		HTTP_Request::POST    => HTTPRequest::METH_POST,
		HTTP_Request::PUT     => HTTPRequest::METH_PUT,
		HTTP_Request::DELETE  => HTTPRequest::METH_DELETE,
		HTTP_Request::OPTIONS => HTTPRequest::METH_OPTIONS,
		HTTP_Request::TRACE   => HTTPRequest::METH_TRACE,
		HTTP_Request::CONNECT => HTTPRequest::METH_CONNECT,
	);
	// Create an http request object
	$http_request = new HTTPRequest($request->uri(), $http_method_mapping[$request->method()]);
	if ($this->_options)
	{
		// Set custom options
		$http_request->setOptions($this->_options);
	}
	// Set headers
	$http_request->setHeaders($request->headers()->getArrayCopy());
	// Set cookies
	$http_request->setCookies($request->cookie());
	// Set query data (?foo=bar&bar=foo)
	$http_request->setQueryData($request->query());
	// Set the body
	if ($request->method() == HTTP_Request::PUT)
	{
		$http_request->addPutData($request->body());
	}
	else
	{
		$http_request->setBody($request->body());
	}
	try
	{
		$http_request->send();
	}
	catch (HTTPRequestException $e)
	{
		throw new Request_Exception($e->getMessage());
	}
	catch (HTTPMalformedHeaderException $e)
	{
		throw new Request_Exception($e->getMessage());
	}
	catch (HTTPEncodingException $e)
	{
		throw new Request_Exception($e->getMessage());
	}
	// Build the response
	$response->status($http_request->getResponseCode())
		->headers($http_request->getResponseHeader())
		->cookie($http_request->getResponseCookies())
		->body($http_request->getResponseBody());
	return $response;
}Processes the request, executing the controller action that handles this request, determined by the Route.
By default, the output from the controller is captured and returned, and no headers are sent.
$request->execute();
Request
$request
required - A request objectResponse
$response
required - A response objectResponsepublic function execute_request(Request $request, Response $response)
{
	if (Kohana::$profiling)
	{
		// Set the benchmark name
		$benchmark = '"'.$request->uri().'"';
		if ($request !== Request::$initial AND Request::$current)
		{
			// Add the parent request uri
			$benchmark .= ' « "'.Request::$current->uri().'"';
		}
		// Start benchmarking
		$benchmark = Profiler::start('Requests', $benchmark);
	}
	// Store the current active request and replace current with new request
	$previous = Request::$current;
	Request::$current = $request;
	// Resolve the POST fields
	if ($post = $request->post())
	{
		$request->body(http_build_query($post, NULL, '&'))
			->headers('content-type', 'application/x-www-form-urlencoded; charset='.Kohana::$charset);
	}
	// If Kohana expose, set the user-agent
	if (Kohana::$expose)
	{
		$request->headers('user-agent', Kohana::version());
	}
	try
	{
		$response = $this->_send_message($request, $response);
	}
	catch (Exception $e)
	{
		// Restore the previous request
		Request::$current = $previous;
		if (isset($benchmark))
		{
			// Delete the benchmark, it is invalid
			Profiler::delete($benchmark);
		}
		// Re-throw the exception
		throw $e;
	}
	// Restore the previous request
	Request::$current = $previous;
	if (isset($benchmark))
	{
		// Stop the benchmark
		Profiler::stop($benchmark);
	}
	// Return the response
	return $response;
}Factory method to create a new Request_Client_External object based on the client name passed, or defaulting to Request_Client_External::$client by default.
Request_Client_External::$client can be set in the application bootstrap.
array
$params
 = array(0)  - Parameters to pass to the clientstring
$client
 = NULL - External client to useRequest_Client_Externalpublic static function factory(array $params = array(), $client = NULL)
{
	if ($client === NULL)
	{
		$client = Request_Client_External::$client;
	}
	$client = new $client($params);
	if ( ! $client instanceof Request_Client_External)
	{
		throw new Request_Exception('Selected client is not a Request_Client_External object.');
	}
	return $client;
}Set and get options for this request.
mixed
$key
 = NULL - Option name, or array of optionsmixed
$value
 = NULL - Option valuemixedRequest_Client_Externalpublic function options($key = NULL, $value = NULL)
{
	if ($key === NULL)
		return $this->_options;
	if (is_array($key))
	{
		$this->_options = $key;
	}
	elseif ($value === NULL)
	{
		return Arr::get($this->_options, $key);
	}
	else
	{
		$this->_options[$key] = $value;
	}
	return $this;
}Assigns the properties of the current Request_Client to another Request_Client instance - used when setting up a subsequent request.
Request_Client
$client
required - $clientpublic function assign_client_properties(Request_Client $client)
{
	$client->cache($this->cache());
	$client->follow($this->follow());
	$client->follow_headers($this->follow_headers());
	$client->header_callbacks($this->header_callbacks());
	$client->max_callback_depth($this->max_callback_depth());
	$client->callback_params($this->callback_params());
}Getter and setter for the internal caching engine, used to cache responses if available and valid.
HTTP_Cache
$cache
 = NULL - Engine to use for cachingHTTP_CacheRequest_Clientpublic function cache(HTTP_Cache $cache = NULL)
{
	if ($cache === NULL)
		return $this->_cache;
	$this->_cache = $cache;
	return $this;
}Getter/Setter for the callback depth property, which is used to track how many recursions have been executed within the current request execution.
int
$depth
 = NULL - Current recursion depthRequest_Client|intpublic function callback_depth($depth = NULL)
{
	if ($depth === NULL)
		return $this->_callback_depth;
	$this->_callback_depth = $depth;
	return $this;
}Getter/Setter for the callback_params array, which allows additional application-specific parameters to be shared with callbacks.
As with other Kohana setter/getters, usage is:
// Set full array
$client->callback_params(array('foo'=>'bar'));
// Set single key
$client->callback_params('foo','bar');
// Get full array
$params = $client->callback_params();
// Get single key
$foo = $client->callback_params('foo');
string|array
$param
 = NULL - $parammixed
$value
 = NULL - $valueRequest_Client|mixedpublic function callback_params($param = NULL, $value = NULL)
{
	// Getter for full array
	if ($param === NULL)
		return $this->_callback_params;
	// Setter for full array
	if (is_array($param))
	{
		$this->_callback_params = $param;
		return $this;
	}
	// Getter for single value
	elseif ($value === NULL)
	{
		return Arr::get($this->_callback_params, $param);
	}
	// Setter for single value
	else
	{
		$this->_callback_params[$param] = $value;
		return $this;
	}
}Processes the request, executing the controller action that handles this request, determined by the Route.
By default, the output from the controller is captured and returned, and no headers are sent.
$request->execute();
Request
$request
required - $requestResponsepublic function execute(Request $request)
{
	// Prevent too much recursion of header callback requests
	if ($this->callback_depth() > $this->max_callback_depth())
		throw new Request_Client_Recursion_Exception(
				"Could not execute request to :uri - too many recursions after :depth requests",
				array(
					':uri' => $request->uri(),
					':depth' => $this->callback_depth() - 1,
				));
	// Execute the request and pass the currently used protocol
	$orig_response = $response = Response::factory(array('_protocol' => $request->protocol()));
	if (($cache = $this->cache()) instanceof HTTP_Cache)
		return $cache->execute($this, $request, $response);
	$response = $this->execute_request($request, $response);
	// Execute response callbacks
	foreach ($this->header_callbacks() as $header => $callback)
	{
		if ($response->headers($header))
		{
			$cb_result = call_user_func($callback, $request, $response, $this);
			if ($cb_result instanceof Request)
			{
				// If the callback returns a request, automatically assign client params
				$this->assign_client_properties($cb_result->client());
				$cb_result->client()->callback_depth($this->callback_depth() + 1);
				// Execute the request
				$response = $cb_result->execute();
			}
			elseif ($cb_result instanceof Response)
			{
				// Assign the returned response
				$response = $cb_result;
			}
			// If the callback has created a new response, do not process any further
			if ($response !== $orig_response)
				break;
		}
	}
	return $response;
}Getter and setter for the follow redirects setting.
bool
$follow
 = NULL - Boolean indicating if redirects should be followedboolRequest_Clientpublic function follow($follow = NULL)
{
	if ($follow === NULL)
		return $this->_follow;
	$this->_follow = $follow;
	return $this;
}Getter and setter for the follow redirects headers array.
array
$follow_headers
 = NULL - Array of headers to be re-used when following a Location headerarrayRequest_Clientpublic function follow_headers($follow_headers = NULL)
{
	if ($follow_headers === NULL)
		return $this->_follow_headers;
	$this->_follow_headers = array_map('strtolower', $follow_headers);
	return $this;
}Getter and setter for the header callbacks array.
Accepts an array with HTTP response headers as keys and a PHP callback function as values. These callbacks will be triggered if a response contains the given header and can either issue a subsequent request or manipulate the response as required.
By default, the Request_Client::on_header_location callback is assigned to the Location header to support automatic redirect following.
$client->header_callbacks(array(
    'Location' => 'Request_Client::on_header_location',
    'WWW-Authenticate' => function($request, $response, $client) {return $new_response;},
);
array
$header_callbacks
 = NULL - Array of callbacks to trigger on presence of given headersRequest_Clientpublic function header_callbacks($header_callbacks = NULL)
{
	if ($header_callbacks === NULL)
		return $this->_header_callbacks;
	$this->_header_callbacks = $header_callbacks;
	return $this;
}Getter and setter for the maximum callback depth property.
This protects the main execution from recursive callback execution (eg following infinite redirects, conflicts between callbacks causing loops etc). Requests will only be allowed to nest to the level set by this param before execution is aborted with a Request_Client_Recursion_Exception.
int
$depth
 = NULL - Maximum number of callback requests to execute before abortingRequest_Client|intpublic function max_callback_depth($depth = NULL)
{
	if ($depth === NULL)
		return $this->_max_callback_depth;
	$this->_max_callback_depth = $depth;
	return $this;
}The default handler for following redirects, triggered by the presence of a Location header in the response.
The client's follow property must be set TRUE and the HTTP response status one of 201, 301, 302, 303 or 307 for the redirect to be followed.
Request
$request
required - $requestResponse
$response
required - $responseRequest_Client
$client
required - $clientpublic static function on_header_location(Request $request, Response $response, Request_Client $client)
{
	// Do we need to follow a Location header ?
	if ($client->follow() AND in_array($response->status(), array(201, 301, 302, 303, 307)))
	{
		// Figure out which method to use for the follow request
		switch ($response->status())
		{
			default:
			case 301:
			case 307:
				$follow_method = $request->method();
				break;
			case 201:
			case 303:
				$follow_method = Request::GET;
				break;
			case 302:
				// Cater for sites with broken HTTP redirect implementations
				if ($client->strict_redirect())
				{
					$follow_method = $request->method();
				}
				else
				{
					$follow_method = Request::GET;
				}
				break;
		}
		// Prepare the additional request, copying any follow_headers that were present on the original request
		$orig_headers = $request->headers()->getArrayCopy();
		$follow_header_keys = array_intersect(array_keys($orig_headers), $client->follow_headers());
		$follow_headers = \Arr::extract($orig_headers, $follow_header_keys);
		$follow_request = Request::factory($response->headers('Location'))
		                         ->method($follow_method)
		                         ->headers($follow_headers);
		if ($follow_method !== Request::GET)
		{
			$follow_request->body($request->body());
		}
		return $follow_request;
	}
	return NULL;
}Getter and setter for the strict redirects setting
HTTP/1.1 specifies that a 302 redirect should be followed using the original request method. However, the vast majority of clients and servers get this wrong, with 302 widely used for 'POST - 302 redirect - GET' patterns. By default, Kohana's client is fully compliant with the HTTP spec. Some non-compliant third party sites may require that strict_redirect is set FALSE to force the client to switch to GET following a 302 response.
bool
$strict_redirect
 = NULL - Boolean indicating if 302 redirects should be followed with the original methodRequest_Clientpublic function strict_redirect($strict_redirect = NULL)
{
	if ($strict_redirect === NULL)
		return $this->_strict_redirect;
	$this->_strict_redirect = $strict_redirect;
	return $this;
}