Files
2026-04-28 15:13:50 +02:00

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;
}
}