182 lines
4.9 KiB
PHP
Raw Normal View History

2013-07-16 19:54:44 -04:00
<?php
namespace PicoFeed\Client;
2013-07-16 19:54:44 -04:00
use PicoFeed\Logging\Logger;
2013-07-16 19:54:44 -04:00
2014-03-16 21:35:57 -04:00
/**
* Stream context HTTP client
*
* @author Frederic Guillot
* @package Client
2014-03-16 21:35:57 -04:00
*/
2014-05-20 14:20:27 -04:00
class Stream extends Client
2013-07-16 19:54:44 -04:00
{
2014-03-16 21:35:57 -04:00
/**
2014-10-19 14:42:31 -04:00
* Prepare HTTP headers
2014-03-16 21:35:57 -04:00
*
2014-10-19 14:42:31 -04:00
* @access private
* @return string[]
2014-03-16 21:35:57 -04:00
*/
2014-10-19 14:42:31 -04:00
private function prepareHeaders()
2013-07-16 19:54:44 -04:00
{
$headers = array(
'Connection: close',
'User-Agent: '.$this->user_agent,
);
2014-05-20 14:20:27 -04:00
if (function_exists('gzdecode')) {
$headers[] = 'Accept-Encoding: gzip';
}
if ($this->etag) {
$headers[] = 'If-None-Match: '.$this->etag;
}
if ($this->last_modified) {
$headers[] = 'If-Modified-Since: '.$this->last_modified;
}
2013-07-16 19:54:44 -04:00
2014-10-19 14:42:31 -04:00
if ($this->proxy_username) {
$headers[] = 'Proxy-Authorization: Basic '.base64_encode($this->proxy_username.':'.$this->proxy_password);
}
2014-12-24 13:50:20 -05:00
if ($this->username && $this->password) {
$headers[] = 'Authorization: Basic '.base64_encode($this->username.':'.$this->password);
}
2014-10-19 14:42:31 -04:00
return $headers;
}
/**
* Prepare stream context
*
* @access private
* @return array
*/
private function prepareContext()
{
$context = array(
2013-07-16 19:54:44 -04:00
'http' => array(
'method' => 'GET',
'protocol_version' => 1.1,
'timeout' => $this->timeout,
2015-01-27 20:13:16 -05:00
'follow_location' => 0,
2013-07-16 19:54:44 -04:00
)
);
2014-05-20 14:20:27 -04:00
if ($this->proxy_hostname) {
Logger::setMessage(get_called_class().' Proxy: '.$this->proxy_hostname.':'.$this->proxy_port);
2014-05-20 14:20:27 -04:00
2014-10-19 14:42:31 -04:00
$context['http']['proxy'] = 'tcp://'.$this->proxy_hostname.':'.$this->proxy_port;
$context['http']['request_fulluri'] = true;
2013-09-23 19:22:13 -04:00
2014-05-20 14:20:27 -04:00
if ($this->proxy_username) {
Logger::setMessage(get_called_class().' Proxy credentials: Yes');
2013-09-23 19:22:13 -04:00
}
2014-05-20 14:20:27 -04:00
else {
Logger::setMessage(get_called_class().' Proxy credentials: No');
2014-05-20 14:20:27 -04:00
}
2013-09-23 19:22:13 -04:00
}
2014-10-19 14:42:31 -04:00
$context['http']['header'] = implode("\r\n", $this->prepareHeaders());
return $context;
}
/**
* Do the HTTP request
*
* @access public
2015-01-27 20:13:16 -05:00
* @param bool $follow_location Flag used when there is an open_basedir restriction
* @return array HTTP response ['body' => ..., 'status' => ..., 'headers' => ...]
2014-10-19 14:42:31 -04:00
*/
2015-01-27 20:13:16 -05:00
public function doRequest($follow_location = false)
2014-10-19 14:42:31 -04:00
{
// Create context
$context = stream_context_create($this->prepareContext());
2013-07-16 19:54:44 -04:00
// Make HTTP request
$stream = @fopen($this->url, 'r', false, $context);
2014-10-19 14:42:31 -04:00
if (! is_resource($stream)) {
throw new InvalidUrlException('Unable to establish a connection');
2014-10-19 14:42:31 -04:00
}
2013-07-16 19:54:44 -04:00
// Get the entire body until the max size
$body = stream_get_contents($stream, $this->max_body_size + 1);
// If the body size is too large abort everything
2014-10-19 14:42:31 -04:00
if (strlen($body) > $this->max_body_size) {
throw new MaxSizeException('Content size too large');
2014-10-19 14:42:31 -04:00
}
2013-07-16 19:54:44 -04:00
// Get HTTP headers response
$metadata = stream_get_meta_data($stream);
if ($metadata['timed_out']) {
throw new TimeoutException('Operation timeout');
}
list($status, $headers) = HttpHeaders::parse($metadata['wrapper_data']);
2013-07-16 19:54:44 -04:00
fclose($stream);
2015-01-27 20:13:16 -05:00
// Do redirect manual to get only the headers of the last request and
// the final url
if ($status == 301 || $status == 302) {
return $this->handleRedirection($headers['Location']);
}
2014-10-19 14:42:31 -04:00
return array(
'status' => $status,
'body' => $this->decodeBody($body, $headers),
'headers' => $headers
);
}
/**
* Decode body response according to the HTTP headers
*
* @access public
* @param string $body Raw body
* @param HttpHeaders $headers HTTP headers
2014-10-19 14:42:31 -04:00
* @return string
*/
public function decodeBody($body, HttpHeaders $headers)
2014-10-19 14:42:31 -04:00
{
if (isset($headers['Transfer-Encoding']) && $headers['Transfer-Encoding'] === 'chunked') {
$body = $this->decodeChunked($body);
}
2014-03-16 21:35:57 -04:00
if (isset($headers['Content-Encoding']) && $headers['Content-Encoding'] === 'gzip') {
$body = @gzdecode($body);
2014-03-16 21:35:57 -04:00
}
2014-10-19 14:42:31 -04:00
return $body;
2013-07-16 19:54:44 -04:00
}
2014-03-16 21:35:57 -04:00
/**
* Decode a chunked body
*
* @access public
* @param string $str Raw body
* @return string Decoded body
*/
public function decodeChunked($str)
{
for ($result = ''; ! empty($str); $str = trim($str)) {
// Get the chunk length
$pos = strpos($str, "\r\n");
$len = hexdec(substr($str, 0, $pos));
// Append the chunk to the result
$result .= substr($str, $pos + 2, $len);
$str = substr($str, $pos + 2 + $len);
}
return $result;
}
2014-03-16 21:35:57 -04:00
}