_stream = null; $this->_context = null; $this->_contextOptions = []; $this->_sslContextOptions = []; $this->_connectionErrors = []; $this->_buildContext($request, $options); return $this->_send($request); } /** * Create the response list based on the headers & content * * Creates one or many response objects based on the number * of redirects that occurred. * * @param array $headers The list of headers from the request(s) * @param string $content The response content. * @return \Cake\Http\Client\Response[] The list of responses from the request(s) */ public function createResponses($headers, $content) { $indexes = $responses = []; foreach ($headers as $i => $header) { if (strtoupper(substr($header, 0, 5)) === 'HTTP/') { $indexes[] = $i; } } $last = count($indexes) - 1; foreach ($indexes as $i => $start) { $end = isset($indexes[$i + 1]) ? $indexes[$i + 1] - $start : null; $headerSlice = array_slice($headers, $start, $end); $body = $i == $last ? $content : ''; $responses[] = $this->_buildResponse($headerSlice, $body); } return $responses; } /** * Build the stream context out of the request object. * * @param \Cake\Http\Client\Request $request The request to build context from. * @param array $options Additional request options. * @return void */ protected function _buildContext(Request $request, $options) { $this->_buildContent($request, $options); $this->_buildHeaders($request, $options); $this->_buildOptions($request, $options); $url = $request->getUri(); $scheme = parse_url($url, PHP_URL_SCHEME); if ($scheme === 'https') { $this->_buildSslContext($request, $options); } $this->_context = stream_context_create([ 'http' => $this->_contextOptions, 'ssl' => $this->_sslContextOptions, ]); } /** * Build the header context for the request. * * Creates cookies & headers. * * @param \Cake\Http\Client\Request $request The request being sent. * @param array $options Array of options to use. * @return void */ protected function _buildHeaders(Request $request, $options) { $headers = []; foreach ($request->getHeaders() as $name => $values) { $headers[] = sprintf('%s: %s', $name, implode(', ', $values)); } $this->_contextOptions['header'] = implode("\r\n", $headers); } /** * Builds the request content based on the request object. * * If the $request->body() is a string, it will be used as is. * Array data will be processed with Cake\Http\Client\FormData * * @param \Cake\Http\Client\Request $request The request being sent. * @param array $options Array of options to use. * @return void */ protected function _buildContent(Request $request, $options) { $body = $request->getBody(); if (empty($body)) { $this->_contextOptions['content'] = ''; return; } $body->rewind(); $this->_contextOptions['content'] = $body->getContents(); } /** * Build miscellaneous options for the request. * * @param \Cake\Http\Client\Request $request The request being sent. * @param array $options Array of options to use. * @return void */ protected function _buildOptions(Request $request, $options) { $this->_contextOptions['method'] = $request->getMethod(); $this->_contextOptions['protocol_version'] = $request->getProtocolVersion(); $this->_contextOptions['ignore_errors'] = true; if (isset($options['timeout'])) { $this->_contextOptions['timeout'] = $options['timeout']; } // Redirects are handled in the client layer because of cookie handling issues. $this->_contextOptions['max_redirects'] = 0; if (isset($options['proxy']['proxy'])) { $this->_contextOptions['request_fulluri'] = true; $this->_contextOptions['proxy'] = $options['proxy']['proxy']; } } /** * Build SSL options for the request. * * @param \Cake\Http\Client\Request $request The request being sent. * @param array $options Array of options to use. * @return void */ protected function _buildSslContext(Request $request, $options) { $sslOptions = [ 'ssl_verify_peer', 'ssl_verify_peer_name', 'ssl_verify_depth', 'ssl_allow_self_signed', 'ssl_cafile', 'ssl_local_cert', 'ssl_passphrase', ]; if (empty($options['ssl_cafile'])) { $options['ssl_cafile'] = CORE_PATH . 'config' . DIRECTORY_SEPARATOR . 'cacert.pem'; } if (!empty($options['ssl_verify_host'])) { $url = $request->getUri(); $host = parse_url($url, PHP_URL_HOST); $this->_sslContextOptions['peer_name'] = $host; } foreach ($sslOptions as $key) { if (isset($options[$key])) { $name = substr($key, 4); $this->_sslContextOptions[$name] = $options[$key]; } } } /** * Open the stream and send the request. * * @param \Cake\Http\Client\Request $request The request object. * @return array Array of populated Response objects * @throws \Cake\Network\Exception\HttpException */ protected function _send(Request $request) { $deadline = false; if (isset($this->_contextOptions['timeout']) && $this->_contextOptions['timeout'] > 0) { $deadline = time() + $this->_contextOptions['timeout']; } $url = $request->getUri(); $this->_open($url); $content = ''; $timedOut = false; while (!feof($this->_stream)) { if ($deadline !== false) { stream_set_timeout($this->_stream, max($deadline - time(), 1)); } $content .= fread($this->_stream, 8192); $meta = stream_get_meta_data($this->_stream); if ($meta['timed_out'] || ($deadline !== false && time() > $deadline)) { $timedOut = true; break; } } $meta = stream_get_meta_data($this->_stream); fclose($this->_stream); if ($timedOut) { throw new HttpException('Connection timed out ' . $url, 504); } $headers = $meta['wrapper_data']; if (isset($headers['headers']) && is_array($headers['headers'])) { $headers = $headers['headers']; } return $this->createResponses($headers, $content); } /** * Build a response object * * @param array $headers Unparsed headers. * @param string $body The response body. * * @return \Cake\Http\Client\Response */ protected function _buildResponse($headers, $body) { return new Response($headers, $body); } /** * Open the socket and handle any connection errors. * * @param string $url The url to connect to. * @return void * @throws \Cake\Core\Exception\Exception */ protected function _open($url) { set_error_handler(function ($code, $message) { $this->_connectionErrors[] = $message; }); $this->_stream = fopen($url, 'rb', false, $this->_context); restore_error_handler(); if (!$this->_stream || !empty($this->_connectionErrors)) { throw new Exception(implode("\n", $this->_connectionErrors)); } } /** * Get the context options * * Useful for debugging and testing context creation. * * @return array */ public function contextOptions() { return array_merge($this->_contextOptions, $this->_sslContextOptions); } } // @deprecated Add backwards compat alias. class_alias('Cake\Http\Client\Adapter\Stream', 'Cake\Network\Http\Adapter\Stream');