*/ require_once 'Google_CacheParser.php'; class Google_HttpStreamIO extends Google_IO { private static $ENTITY_HTTP_METHODS = array("POST" => null, "PUT" => null); private static $DEFAULT_HTTP_CONTEXT = array( "follow_location" => 0, "ignore_errors" => 1, ); private static $DEFAULT_SSL_CONTEXT = array( "verify_peer" => true, ); /** * Perform an authenticated / signed apiHttpRequest. * This function takes the apiHttpRequest, calls apiAuth->sign on it * (which can modify the request in what ever way fits the auth mechanism) * and then calls Google_HttpStreamIO::makeRequest on the signed request * * @param Google_HttpRequest $request * @return Google_HttpRequest The resulting HTTP response including the * responseHttpCode, responseHeaders and responseBody. */ public function authenticatedRequest(Google_HttpRequest $request) { $request = Google_Client::$auth->sign($request); return $this->makeRequest($request); } /** * Execute a apiHttpRequest * * @param Google_HttpRequest $request the http request to be executed * @return Google_HttpRequest http request with the response http code, * response headers and response body filled in * @throws Google_IOException on curl or IO error */ public function makeRequest(Google_HttpRequest $request) { // First, check to see if we have a valid cached version. $cached = $this->getCachedRequest($request); if ($cached !== false) { if (!$this->checkMustRevaliadateCachedRequest($cached, $request)) { return $cached; } } $default_options = stream_context_get_options(stream_context_get_default()); $requestHttpContext = array_key_exists('http', $default_options) ? $default_options['http'] : array(); if (array_key_exists($request->getRequestMethod(), self::$ENTITY_HTTP_METHODS)) { $request = $this->processEntityRequest($request); } if ($request->getPostBody()) { $requestHttpContext["content"] = $request->getPostBody(); } $requestHeaders = $request->getRequestHeaders(); if ($requestHeaders && is_array($requestHeaders)) { $headers = ""; foreach($requestHeaders as $k => $v) { $headers .= "$k: $v\n"; } $requestHttpContext["header"] = $headers; } $requestHttpContext["method"] = $request->getRequestMethod(); $requestHttpContext["user_agent"] = $request->getUserAgent(); $requestSslContext = array_key_exists('ssl', $default_options) ? $default_options['ssl'] : array(); if (!array_key_exists("cafile", $requestSslContext)) { $requestSslContext["cafile"] = dirname(__FILE__) . '/cacerts.pem'; } $options = array("http" => array_merge(self::$DEFAULT_HTTP_CONTEXT, $requestHttpContext), "ssl" => array_merge(self::$DEFAULT_SSL_CONTEXT, $requestSslContext)); $context = stream_context_create($options); $response_data = file_get_contents($request->getUrl(), false, $context); if (false === $response_data) { throw new Google_IOException("HTTP Error: Unable to connect"); } $respHttpCode = $this->getHttpResponseCode($http_response_header); $responseHeaders = $this->getHttpResponseHeaders($http_response_header); if ($respHttpCode == 304 && $cached) { // If the server responded NOT_MODIFIED, return the cached request. $this->updateCachedRequest($cached, $responseHeaders); return $cached; } $request->setResponseHttpCode($respHttpCode); $request->setResponseHeaders($responseHeaders); $request->setResponseBody($response_data); // Store the request in cache (the function checks to see if the request // can actually be cached) $this->setCachedRequest($request); return $request; } /** * Set options that update the transport implementation's behavior. * @param $options */ public function setOptions($options) { } private function getHttpResponseCode($response_headers) { $header_count = count($response_headers); for ($i = 0; $i < $header_count; $i++) { $header = $response_headers[$i]; if (strncasecmp("HTTP", $header, strlen("HTTP")) == 0) { $response = explode(' ', $header); return $response[1]; } } return 'UNKNOWN'; } private function getHttpResponseHeaders($response_headers) { $header_count = count($response_headers); $headers = array(); for ($i = 0; $i < $header_count; $i++) { $header = $response_headers[$i]; $header_parts = explode(':', $header); if (count($header_parts) == 2) { $headers[$header_parts[0]] = $header_parts[1]; } } return $headers; } }