202 lines
5.9 KiB
PHP
202 lines
5.9 KiB
PHP
<?php
|
|
/**
|
|
* Copyright (c) 2014-present, Facebook, Inc. All rights reserved.
|
|
*
|
|
* You are hereby granted a non-exclusive, worldwide, royalty-free license to
|
|
* use, copy, modify, and distribute this software in source code or binary
|
|
* form for use in connection with the web services and APIs provided by
|
|
* Facebook.
|
|
*
|
|
* As with any software that integrates with the Facebook platform, your use
|
|
* of this software is subject to the Facebook Developer Principles and
|
|
* Policies [http://developers.facebook.com/policy/]. This copyright notice
|
|
* shall be included in all copies or substantial portions of the software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
* DEALINGS IN THE SOFTWARE.
|
|
*
|
|
*/
|
|
|
|
namespace FacebookPixelPlugin\FacebookAds\Http\Adapter;
|
|
|
|
use FacebookPixelPlugin\FacebookAds\Exception\Exception;
|
|
use FacebookPixelPlugin\FacebookAds\Http\Adapter\Curl\AbstractCurl;
|
|
use FacebookPixelPlugin\FacebookAds\Http\Adapter\Curl\Curl;
|
|
use FacebookPixelPlugin\FacebookAds\Http\Adapter\Curl\CurlInterface;
|
|
use FacebookPixelPlugin\FacebookAds\Http\Client;
|
|
use FacebookPixelPlugin\FacebookAds\Http\Headers;
|
|
use FacebookPixelPlugin\FacebookAds\Http\RequestInterface;
|
|
use FacebookPixelPlugin\FacebookAds\Http\ResponseInterface;
|
|
|
|
class CurlAdapter extends AbstractAdapter {
|
|
|
|
/**
|
|
* @var CurlInterface
|
|
*/
|
|
protected $curl;
|
|
|
|
/**
|
|
* @var \ArrayObject
|
|
*/
|
|
protected $opts;
|
|
|
|
/**
|
|
* @param Client $client
|
|
* @param CurlInterface $curl
|
|
*/
|
|
public function __construct(Client $client, ?CurlInterface $curl = null) {
|
|
parent::__construct($client);
|
|
$this->curl = $curl ?: AbstractCurl::createOptimalVersion();
|
|
$this->curl->init();
|
|
}
|
|
|
|
/**
|
|
* @return Curl
|
|
*/
|
|
public function getCurl() {
|
|
return $this->curl;
|
|
}
|
|
|
|
/**
|
|
* @return \ArrayObject
|
|
*/
|
|
public function getOpts() {
|
|
if ($this->opts === null) {
|
|
$this->opts = new \ArrayObject(array(
|
|
CURLOPT_CONNECTTIMEOUT => 10,
|
|
CURLOPT_TIMEOUT => 60,
|
|
CURLOPT_RETURNTRANSFER => true,
|
|
CURLOPT_HEADER => true,
|
|
CURLOPT_CAINFO => $this->getCaBundlePath(),
|
|
));
|
|
}
|
|
|
|
return $this->opts;
|
|
}
|
|
|
|
/**
|
|
* @param \ArrayObject $opts
|
|
*/
|
|
public function setOpts(\ArrayObject $opts) {
|
|
$this->opts = $opts;
|
|
}
|
|
|
|
/**
|
|
* @return int
|
|
*/
|
|
protected function getheaderSize() {
|
|
return $this->getCurl()->getInfo(CURLINFO_HEADER_SIZE) ?? 0;
|
|
}
|
|
|
|
/**
|
|
* Extracts the headers and the body into a two-part array
|
|
* @param string $raw_response
|
|
* @return array
|
|
*/
|
|
protected function extractResponseHeadersAndBody($raw_response) {
|
|
$header_size = $this->getheaderSize();
|
|
|
|
$raw_headers = mb_substr($raw_response, 0, $header_size);
|
|
$raw_body = mb_substr($raw_response, $header_size);
|
|
|
|
return array(trim($raw_headers), trim($raw_body));
|
|
}
|
|
|
|
/**
|
|
* @param Headers $headers
|
|
* @param string $raw_headers
|
|
*/
|
|
protected function parseHeaders(Headers $headers, $raw_headers) {
|
|
$raw_headers = str_replace("\r\n", "\n", $raw_headers);
|
|
|
|
// There will be multiple headers if a 301 was followed
|
|
// or a proxy was followed, etc
|
|
$header_collection = explode("\n\n", trim($raw_headers));
|
|
// We just want the last response (at the end)
|
|
$raw_headers = array_pop($header_collection);
|
|
|
|
$header_components = explode("\n", $raw_headers);
|
|
foreach ($header_components as $line) {
|
|
if (strpos($line, ': ') === false) {
|
|
$headers['http_code'] = $line;
|
|
} else {
|
|
list ($key, $value) = explode(': ', $line, 2);
|
|
$headers[$key] = $value;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param RequestInterface $request
|
|
* @return ResponseInterface
|
|
* @throws Exception
|
|
*/
|
|
public function sendRequest(RequestInterface $request) {
|
|
$response = $this->getClient()->createResponse();
|
|
$this->getCurl()->reset();
|
|
$curlopts = array(
|
|
CURLOPT_URL => $request->getUrl(),
|
|
);
|
|
|
|
$method = $request->getMethod();
|
|
if ($method !== RequestInterface::METHOD_GET
|
|
&& $method !== RequestInterface::METHOD_POST) {
|
|
$curlopts[CURLOPT_CUSTOMREQUEST] = $method;
|
|
}
|
|
|
|
$curlopts = $this->getOpts()->getArrayCopy() + $curlopts;
|
|
|
|
if ($request->getHeaders()->count()) {
|
|
$headers = array();
|
|
foreach ($request->getHeaders() as $header => $value) {
|
|
$headers[] = "{$header}: {$value}";
|
|
}
|
|
$curlopts[CURLOPT_HTTPHEADER] = $headers;
|
|
}
|
|
|
|
$postfields = array();
|
|
if ($method === RequestInterface::METHOD_POST
|
|
&& $request->getFileParams()->count()
|
|
) {
|
|
$postfields = array_merge(
|
|
$postfields,
|
|
array_map(
|
|
array($this->getCurl(), 'preparePostFileField'),
|
|
$request->getFileParams()->getArrayCopy()));
|
|
}
|
|
if ($method !== RequestInterface::METHOD_GET
|
|
&& $request->getBodyParams()->count()) {
|
|
$postfields
|
|
= array_merge($postfields, $request->getBodyParams()->export());
|
|
}
|
|
|
|
if (!empty($postfields)) {
|
|
$curlopts[CURLOPT_POSTFIELDS] = $postfields;
|
|
}
|
|
|
|
$this->getCurl()->setoptArray($curlopts);
|
|
$raw_response = $this->getCurl()->exec();
|
|
|
|
$status_code = $this->getCurl()->getInfo(CURLINFO_HTTP_CODE);
|
|
$curl_errno = $this->getCurl()->errno();
|
|
$curl_error = $curl_errno ? $this->getCurl()->error() : null;
|
|
|
|
$response_parts = $this->extractResponseHeadersAndBody($raw_response);
|
|
|
|
$response->setStatusCode($status_code);
|
|
$this->parseHeaders($response->getHeaders(), $response_parts[0]);
|
|
$response->setBody($response_parts[1]);
|
|
|
|
if ($curl_errno) {
|
|
throw new Exception($curl_error, $curl_errno);
|
|
}
|
|
|
|
return $response;
|
|
}
|
|
}
|