File indexing completed on 2025-01-19 05:21:14
0001 <?php 0002 0003 /** 0004 * Zend Framework 0005 * 0006 * LICENSE 0007 * 0008 * This source file is subject to the new BSD license that is bundled 0009 * with this package in the file LICENSE.txt. 0010 * It is also available through the world-wide-web at this URL: 0011 * http://framework.zend.com/license/new-bsd 0012 * If you did not receive a copy of the license and are unable to 0013 * obtain it through the world-wide-web, please send an email 0014 * to license@zend.com so we can send you a copy immediately. 0015 * 0016 * @category Zend 0017 * @package Zend_Http 0018 * @subpackage Client_Adapter 0019 * @version $Id$ 0020 * @copyright Copyright (c) 2005-2015 Zend Technologies USA Inc. (http://www.zend.com) 0021 * @license http://framework.zend.com/license/new-bsd New BSD License 0022 */ 0023 0024 /** 0025 * @see Zend_Uri_Http 0026 */ 0027 // require_once 'Zend/Uri/Http.php'; 0028 0029 /** 0030 * @see Zend_Http_Client_Adapter_Interface 0031 */ 0032 // require_once 'Zend/Http/Client/Adapter/Interface.php'; 0033 /** 0034 * @see Zend_Http_Client_Adapter_Stream 0035 */ 0036 // require_once 'Zend/Http/Client/Adapter/Stream.php'; 0037 0038 /** 0039 * An adapter class for Zend_Http_Client based on the curl extension. 0040 * Curl requires libcurl. See for full requirements the PHP manual: http://php.net/curl 0041 * 0042 * @category Zend 0043 * @package Zend_Http 0044 * @subpackage Client_Adapter 0045 * @copyright Copyright (c) 2005-2015 Zend Technologies USA Inc. (http://www.zend.com) 0046 * @license http://framework.zend.com/license/new-bsd New BSD License 0047 */ 0048 class Zend_Http_Client_Adapter_Curl implements Zend_Http_Client_Adapter_Interface, Zend_Http_Client_Adapter_Stream 0049 { 0050 /** 0051 * Parameters array 0052 * 0053 * @var array 0054 */ 0055 protected $_config = array(); 0056 0057 /** 0058 * What host/port are we connected to? 0059 * 0060 * @var array 0061 */ 0062 protected $_connected_to = array(null, null); 0063 0064 /** 0065 * The curl session handle 0066 * 0067 * @var resource|null 0068 */ 0069 protected $_curl = null; 0070 0071 /** 0072 * List of cURL options that should never be overwritten 0073 * 0074 * @var array 0075 */ 0076 protected $_invalidOverwritableCurlOptions; 0077 0078 /** 0079 * Response gotten from server 0080 * 0081 * @var string 0082 */ 0083 protected $_response = null; 0084 0085 /** 0086 * Stream for storing output 0087 * 0088 * @var resource 0089 */ 0090 protected $out_stream; 0091 0092 /** 0093 * Adapter constructor 0094 * 0095 * Config is set using setConfig() 0096 * 0097 * @return void 0098 * @throws Zend_Http_Client_Adapter_Exception 0099 */ 0100 public function __construct() 0101 { 0102 if (!extension_loaded('curl')) { 0103 // require_once 'Zend/Http/Client/Adapter/Exception.php'; 0104 throw new Zend_Http_Client_Adapter_Exception('cURL extension has to be loaded to use this Zend_Http_Client adapter.'); 0105 } 0106 $this->_invalidOverwritableCurlOptions = array( 0107 CURLOPT_HTTPGET, 0108 CURLOPT_POST, 0109 CURLOPT_PUT, 0110 CURLOPT_CUSTOMREQUEST, 0111 CURLOPT_HEADER, 0112 CURLOPT_RETURNTRANSFER, 0113 CURLOPT_HTTPHEADER, 0114 CURLOPT_POSTFIELDS, 0115 CURLOPT_INFILE, 0116 CURLOPT_INFILESIZE, 0117 CURLOPT_PORT, 0118 CURLOPT_MAXREDIRS, 0119 CURLOPT_CONNECTTIMEOUT, 0120 CURL_HTTP_VERSION_1_1, 0121 CURL_HTTP_VERSION_1_0, 0122 ); 0123 } 0124 0125 /** 0126 * Set the configuration array for the adapter 0127 * 0128 * @throws Zend_Http_Client_Adapter_Exception 0129 * @param Zend_Config | array $config 0130 * @return Zend_Http_Client_Adapter_Curl 0131 */ 0132 public function setConfig($config = array()) 0133 { 0134 if ($config instanceof Zend_Config) { 0135 $config = $config->toArray(); 0136 0137 } elseif (! is_array($config)) { 0138 // require_once 'Zend/Http/Client/Adapter/Exception.php'; 0139 throw new Zend_Http_Client_Adapter_Exception( 0140 'Array or Zend_Config object expected, got ' . gettype($config) 0141 ); 0142 } 0143 0144 if(isset($config['proxy_user']) && isset($config['proxy_pass'])) { 0145 $this->setCurlOption(CURLOPT_PROXYUSERPWD, $config['proxy_user'].":".$config['proxy_pass']); 0146 unset($config['proxy_user'], $config['proxy_pass']); 0147 } 0148 0149 foreach ($config as $k => $v) { 0150 $option = strtolower($k); 0151 switch($option) { 0152 case 'proxy_host': 0153 $this->setCurlOption(CURLOPT_PROXY, $v); 0154 break; 0155 case 'proxy_port': 0156 $this->setCurlOption(CURLOPT_PROXYPORT, $v); 0157 break; 0158 default: 0159 $this->_config[$option] = $v; 0160 break; 0161 } 0162 } 0163 0164 return $this; 0165 } 0166 0167 /** 0168 * Retrieve the array of all configuration options 0169 * 0170 * @return array 0171 */ 0172 public function getConfig() 0173 { 0174 return $this->_config; 0175 } 0176 0177 /** 0178 * Direct setter for cURL adapter related options. 0179 * 0180 * @param string|int $option 0181 * @param mixed $value 0182 * @return Zend_Http_Adapter_Curl 0183 */ 0184 public function setCurlOption($option, $value) 0185 { 0186 if (!isset($this->_config['curloptions'])) { 0187 $this->_config['curloptions'] = array(); 0188 } 0189 $this->_config['curloptions'][$option] = $value; 0190 return $this; 0191 } 0192 0193 /** 0194 * Initialize curl 0195 * 0196 * @param string $host 0197 * @param int $port 0198 * @param boolean $secure 0199 * @return void 0200 * @throws Zend_Http_Client_Adapter_Exception if unable to connect 0201 */ 0202 public function connect($host, $port = 80, $secure = false) 0203 { 0204 // If we're already connected, disconnect first 0205 if ($this->_curl) { 0206 $this->close(); 0207 } 0208 0209 // If we are connected to a different server or port, disconnect first 0210 if ($this->_curl 0211 && is_array($this->_connected_to) 0212 && ($this->_connected_to[0] != $host 0213 || $this->_connected_to[1] != $port) 0214 ) { 0215 $this->close(); 0216 } 0217 0218 // Do the actual connection 0219 $this->_curl = curl_init(); 0220 if ($port != 80) { 0221 curl_setopt($this->_curl, CURLOPT_PORT, intval($port)); 0222 } 0223 0224 // Set connection timeout 0225 $connectTimeout = $this->_config['timeout']; 0226 $constant = CURLOPT_CONNECTTIMEOUT; 0227 if (defined('CURLOPT_CONNECTTIMEOUT_MS')) { 0228 $connectTimeout *= 1000; 0229 $constant = constant('CURLOPT_CONNECTTIMEOUT_MS'); 0230 } 0231 curl_setopt($this->_curl, $constant, $connectTimeout); 0232 0233 // Set request timeout (once connection is established) 0234 if (array_key_exists('request_timeout', $this->_config)) { 0235 $requestTimeout = $this->_config['request_timeout']; 0236 $constant = CURLOPT_TIMEOUT; 0237 if (defined('CURLOPT_TIMEOUT_MS')) { 0238 $requestTimeout *= 1000; 0239 $constant = constant('CURLOPT_TIMEOUT_MS'); 0240 } 0241 curl_setopt($this->_curl, $constant, $requestTimeout); 0242 } 0243 0244 // Set Max redirects 0245 curl_setopt($this->_curl, CURLOPT_MAXREDIRS, $this->_config['maxredirects']); 0246 0247 if (!$this->_curl) { 0248 $this->close(); 0249 0250 // require_once 'Zend/Http/Client/Adapter/Exception.php'; 0251 throw new Zend_Http_Client_Adapter_Exception('Unable to Connect to ' . $host . ':' . $port); 0252 } 0253 0254 if ($secure !== false) { 0255 // Behave the same like Zend_Http_Adapter_Socket on SSL options. 0256 if (isset($this->_config['sslcert'])) { 0257 curl_setopt($this->_curl, CURLOPT_SSLCERT, $this->_config['sslcert']); 0258 } 0259 if (isset($this->_config['sslpassphrase'])) { 0260 curl_setopt($this->_curl, CURLOPT_SSLCERTPASSWD, $this->_config['sslpassphrase']); 0261 } 0262 } 0263 0264 // Update connected_to 0265 $this->_connected_to = array($host, $port); 0266 } 0267 0268 /** 0269 * Send request to the remote server 0270 * 0271 * @param string $method 0272 * @param Zend_Uri_Http $uri 0273 * @param float $http_ver 0274 * @param array $headers 0275 * @param string $body 0276 * @return string $request 0277 * @throws Zend_Http_Client_Adapter_Exception If connection fails, connected to wrong host, no PUT file defined, unsupported method, or unsupported cURL option 0278 */ 0279 public function write($method, $uri, $httpVersion = 1.1, $headers = array(), $body = '') 0280 { 0281 // Make sure we're properly connected 0282 if (!$this->_curl) { 0283 // require_once 'Zend/Http/Client/Adapter/Exception.php'; 0284 throw new Zend_Http_Client_Adapter_Exception("Trying to write but we are not connected"); 0285 } 0286 0287 if ($this->_connected_to[0] != $uri->getHost() || $this->_connected_to[1] != $uri->getPort()) { 0288 // require_once 'Zend/Http/Client/Adapter/Exception.php'; 0289 throw new Zend_Http_Client_Adapter_Exception("Trying to write but we are connected to the wrong host"); 0290 } 0291 0292 // set URL 0293 curl_setopt($this->_curl, CURLOPT_URL, $uri->__toString()); 0294 0295 // ensure correct curl call 0296 $curlValue = true; 0297 switch ($method) { 0298 case Zend_Http_Client::GET: 0299 $curlMethod = CURLOPT_HTTPGET; 0300 break; 0301 0302 case Zend_Http_Client::POST: 0303 $curlMethod = CURLOPT_POST; 0304 break; 0305 0306 case Zend_Http_Client::PUT: 0307 // There are two different types of PUT request, either a Raw Data string has been set 0308 // or CURLOPT_INFILE and CURLOPT_INFILESIZE are used. 0309 if(is_resource($body)) { 0310 $this->_config['curloptions'][CURLOPT_INFILE] = $body; 0311 } 0312 if (isset($this->_config['curloptions'][CURLOPT_INFILE])) { 0313 // Now we will probably already have Content-Length set, so that we have to delete it 0314 // from $headers at this point: 0315 foreach ($headers AS $k => $header) { 0316 if (preg_match('/Content-Length:\s*(\d+)/i', $header, $m)) { 0317 if(is_resource($body)) { 0318 $this->_config['curloptions'][CURLOPT_INFILESIZE] = (int)$m[1]; 0319 } 0320 unset($headers[$k]); 0321 } 0322 } 0323 0324 if (!isset($this->_config['curloptions'][CURLOPT_INFILESIZE])) { 0325 // require_once 'Zend/Http/Client/Adapter/Exception.php'; 0326 throw new Zend_Http_Client_Adapter_Exception("Cannot set a file-handle for cURL option CURLOPT_INFILE without also setting its size in CURLOPT_INFILESIZE."); 0327 } 0328 0329 if(is_resource($body)) { 0330 $body = ''; 0331 } 0332 0333 $curlMethod = CURLOPT_PUT; 0334 } else { 0335 $curlMethod = CURLOPT_CUSTOMREQUEST; 0336 $curlValue = "PUT"; 0337 } 0338 break; 0339 0340 case Zend_Http_Client::PATCH: 0341 $curlMethod = CURLOPT_CUSTOMREQUEST; 0342 $curlValue = "PATCH"; 0343 break; 0344 0345 case Zend_Http_Client::DELETE: 0346 $curlMethod = CURLOPT_CUSTOMREQUEST; 0347 $curlValue = "DELETE"; 0348 break; 0349 0350 case Zend_Http_Client::OPTIONS: 0351 $curlMethod = CURLOPT_CUSTOMREQUEST; 0352 $curlValue = "OPTIONS"; 0353 break; 0354 0355 case Zend_Http_Client::TRACE: 0356 $curlMethod = CURLOPT_CUSTOMREQUEST; 0357 $curlValue = "TRACE"; 0358 break; 0359 0360 case Zend_Http_Client::HEAD: 0361 $curlMethod = CURLOPT_CUSTOMREQUEST; 0362 $curlValue = "HEAD"; 0363 break; 0364 0365 default: 0366 // For now, through an exception for unsupported request methods 0367 // require_once 'Zend/Http/Client/Adapter/Exception.php'; 0368 throw new Zend_Http_Client_Adapter_Exception("Method currently not supported"); 0369 } 0370 0371 if(is_resource($body) && $curlMethod != CURLOPT_PUT) { 0372 // require_once 'Zend/Http/Client/Adapter/Exception.php'; 0373 throw new Zend_Http_Client_Adapter_Exception("Streaming requests are allowed only with PUT"); 0374 } 0375 0376 // get http version to use 0377 $curlHttp = ($httpVersion == 1.1) ? CURL_HTTP_VERSION_1_1 : CURL_HTTP_VERSION_1_0; 0378 0379 // mark as HTTP request and set HTTP method 0380 curl_setopt($this->_curl, CURLOPT_HTTP_VERSION, $curlHttp); 0381 curl_setopt($this->_curl, $curlMethod, $curlValue); 0382 0383 if($this->out_stream) { 0384 // headers will be read into the response 0385 curl_setopt($this->_curl, CURLOPT_HEADER, false); 0386 curl_setopt($this->_curl, CURLOPT_HEADERFUNCTION, array($this, "readHeader")); 0387 // and data will be written into the file 0388 curl_setopt($this->_curl, CURLOPT_FILE, $this->out_stream); 0389 } else { 0390 // ensure headers are also returned 0391 curl_setopt($this->_curl, CURLOPT_HEADER, true); 0392 curl_setopt($this->_curl, CURLINFO_HEADER_OUT, true); 0393 0394 // ensure actual response is returned 0395 curl_setopt($this->_curl, CURLOPT_RETURNTRANSFER, true); 0396 } 0397 0398 // set additional headers 0399 $headers['Accept'] = ''; 0400 curl_setopt($this->_curl, CURLOPT_HTTPHEADER, $headers); 0401 0402 /** 0403 * Make sure POSTFIELDS is set after $curlMethod is set: 0404 * @link http://de2.php.net/manual/en/function.curl-setopt.php#81161 0405 */ 0406 if ($method == Zend_Http_Client::POST) { 0407 curl_setopt($this->_curl, CURLOPT_POSTFIELDS, $body); 0408 } elseif ($curlMethod == CURLOPT_PUT) { 0409 // this covers a PUT by file-handle: 0410 // Make the setting of this options explicit (rather than setting it through the loop following a bit lower) 0411 // to group common functionality together. 0412 curl_setopt($this->_curl, CURLOPT_INFILE, $this->_config['curloptions'][CURLOPT_INFILE]); 0413 curl_setopt($this->_curl, CURLOPT_INFILESIZE, $this->_config['curloptions'][CURLOPT_INFILESIZE]); 0414 unset($this->_config['curloptions'][CURLOPT_INFILE]); 0415 unset($this->_config['curloptions'][CURLOPT_INFILESIZE]); 0416 } elseif ($method == Zend_Http_Client::PUT) { 0417 // This is a PUT by a setRawData string, not by file-handle 0418 curl_setopt($this->_curl, CURLOPT_POSTFIELDS, $body); 0419 } elseif ($method == Zend_Http_Client::PATCH) { 0420 // This is a PATCH by a setRawData string 0421 curl_setopt($this->_curl, CURLOPT_POSTFIELDS, $body); 0422 } elseif ($method == Zend_Http_Client::DELETE) { 0423 // This is a DELETE by a setRawData string 0424 curl_setopt($this->_curl, CURLOPT_POSTFIELDS, $body); 0425 } elseif ($method == Zend_Http_Client::OPTIONS) { 0426 // This is an OPTIONS by a setRawData string 0427 curl_setopt($this->_curl, CURLOPT_POSTFIELDS, $body); 0428 } 0429 0430 // set additional curl options 0431 if (isset($this->_config['curloptions'])) { 0432 foreach ((array)$this->_config['curloptions'] as $k => $v) { 0433 if (!in_array($k, $this->_invalidOverwritableCurlOptions)) { 0434 if (curl_setopt($this->_curl, $k, $v) == false) { 0435 // require_once 'Zend/Http/Client/Exception.php'; 0436 throw new Zend_Http_Client_Exception(sprintf("Unknown or erroreous cURL option '%s' set", $k)); 0437 } 0438 } 0439 } 0440 } 0441 0442 // send the request 0443 $response = curl_exec($this->_curl); 0444 0445 // if we used streaming, headers are already there 0446 if(!is_resource($this->out_stream)) { 0447 $this->_response = $response; 0448 } 0449 0450 $request = curl_getinfo($this->_curl, CURLINFO_HEADER_OUT); 0451 $request .= $body; 0452 0453 if (empty($this->_response)) { 0454 // require_once 'Zend/Http/Client/Exception.php'; 0455 throw new Zend_Http_Client_Exception("Error in cURL request: " . curl_error($this->_curl)); 0456 } 0457 0458 // cURL automatically decodes chunked-messages, this means we have to disallow the Zend_Http_Response to do it again 0459 if (stripos($this->_response, "Transfer-Encoding: chunked\r\n")) { 0460 $this->_response = str_ireplace("Transfer-Encoding: chunked\r\n", '', $this->_response); 0461 } 0462 0463 // Eliminate multiple HTTP responses. 0464 do { 0465 $parts = preg_split('|(?:\r?\n){2}|m', $this->_response, 2); 0466 $again = false; 0467 0468 if (isset($parts[1]) && preg_match("|^HTTP/1\.[01](.*?)\r\n|mi", $parts[1])) { 0469 $this->_response = $parts[1]; 0470 $again = true; 0471 } 0472 } while ($again); 0473 0474 // cURL automatically handles Proxy rewrites, remove the "HTTP/1.0 200 Connection established" string: 0475 if (stripos($this->_response, "HTTP/1.0 200 Connection established\r\n\r\n") !== false) { 0476 $this->_response = str_ireplace("HTTP/1.0 200 Connection established\r\n\r\n", '', $this->_response); 0477 } 0478 0479 return $request; 0480 } 0481 0482 /** 0483 * Return read response from server 0484 * 0485 * @return string 0486 */ 0487 public function read() 0488 { 0489 return $this->_response; 0490 } 0491 0492 /** 0493 * Close the connection to the server 0494 * 0495 */ 0496 public function close() 0497 { 0498 if(is_resource($this->_curl)) { 0499 curl_close($this->_curl); 0500 } 0501 $this->_curl = null; 0502 $this->_connected_to = array(null, null); 0503 } 0504 0505 /** 0506 * Get cUrl Handle 0507 * 0508 * @return resource 0509 */ 0510 public function getHandle() 0511 { 0512 return $this->_curl; 0513 } 0514 0515 /** 0516 * Set output stream for the response 0517 * 0518 * @param resource $stream 0519 * @return Zend_Http_Client_Adapter_Socket 0520 */ 0521 public function setOutputStream($stream) 0522 { 0523 $this->out_stream = $stream; 0524 return $this; 0525 } 0526 0527 /** 0528 * Header reader function for CURL 0529 * 0530 * @param resource $curl 0531 * @param string $header 0532 * @return int 0533 */ 0534 public function readHeader($curl, $header) 0535 { 0536 $this->_response .= $header; 0537 return strlen($header); 0538 } 0539 }