Spaces:
No application file
No application file
namespace MauticPlugin\MauticFullContactBundle\Services; | |
use MauticPlugin\MauticFullContactBundle\Exception\NoCreditException; | |
use MauticPlugin\MauticFullContactBundle\Exception\NotImplementedException; | |
/** | |
* This class handles the actually HTTP request to the FullContact endpoint. | |
* | |
* @author Keith Casey <contrib@caseysoftware.com> | |
* @license http://www.apache.org/licenses/LICENSE-2.0 Apache | |
*/ | |
class FullContact_Base | |
{ | |
public const REQUEST_LATENCY = 0.2; | |
public const USER_AGENT = 'caseysoftware/fullcontact-php-0.9.0'; | |
private \DateTime $_next_req_time; | |
// protected $_baseUri = 'https://requestbin.fullcontact.com/1ailj6d1?'; | |
protected $_baseUri = 'https://api.fullcontact.com/'; | |
protected $_version = 'v2'; | |
protected $_resourceUri = ''; | |
protected $_webhookUrl; | |
protected $_webhookId; | |
protected $_webhookJson = false; | |
protected $_supportedMethods = []; | |
public $response_obj; | |
public $response_code; | |
public $response_json; | |
/** | |
* Slow down calls to the FullContact API if needed. | |
*/ | |
private function _wait_for_rate_limit(): void | |
{ | |
$now = new \DateTime(); | |
if ($this->_next_req_time->getTimestamp() > $now->getTimestamp()) { | |
$t = $this->_next_req_time->getTimestamp() - $now->getTimestamp(); | |
sleep($t); | |
} | |
} | |
/** | |
* @param string $hdr | |
*/ | |
private function _update_rate_limit($hdr): void | |
{ | |
$remaining = (float) $hdr['X-Rate-Limit-Remaining']; | |
$reset = (float) $hdr['X-Rate-Limit-Reset']; | |
$spacing = $reset / (1.0 + $remaining); | |
$delay = $spacing - self::REQUEST_LATENCY; | |
$this->_next_req_time = new \DateTime('now + '.$delay.' seconds'); | |
} | |
/** | |
* The base constructor Sets the API key available from here: | |
* http://fullcontact.com/getkey. | |
* | |
* @param string $_apiKey | |
*/ | |
public function __construct( | |
protected $_apiKey | |
) { | |
$this->_next_req_time = new \DateTime('@0'); | |
} | |
/** | |
* This sets the webhook url for all requests made for this service | |
* instance. To unset, just use setWebhookUrl(null). | |
* | |
* @author David Boskovic <me@david.gs> @dboskovic | |
* | |
* @param string $url | |
* @param string $id | |
* @param bool $json | |
* | |
* @return object | |
*/ | |
public function setWebhookUrl($url, $id = null, $json = false) | |
{ | |
$this->_webhookUrl = $url; | |
$this->_webhookId = $id; | |
$this->_webhookJson = $json; | |
return $this; | |
} | |
/** | |
* This is a pretty close copy of my work on the Contactually PHP library | |
* available here: http://github.com/caseysoftware/contactually-php. | |
* | |
* @author Keith Casey <contrib@caseysoftware.com> | |
* @author David Boskovic <me@david.gs> @dboskovic | |
* | |
* @param array $params | |
* @param array $postData | |
* | |
* @return object | |
* | |
* @throws NoCreditException | |
* @throws NotImplementedException | |
*/ | |
protected function _execute($params = [], $postData = null) | |
{ | |
if (null === $postData && !in_array($params['method'], $this->_supportedMethods, true)) { | |
throw new NotImplementedException(self::class.' does not support the ['.$params['method'].'] method'); | |
} | |
if (array_key_exists('method', $params)) { | |
unset($params['method']); | |
} | |
$this->_wait_for_rate_limit(); | |
$params['apiKey'] = $this->_apiKey; | |
if ($this->_webhookUrl) { | |
$params['webhookUrl'] = $this->_webhookUrl; | |
} | |
if ($this->_webhookId) { | |
$params['webhookId'] = $this->_webhookId; | |
} | |
if ($this->_webhookJson) { | |
$params['webhookBody'] = 'json'; | |
} | |
$fullUrl = $this->_baseUri.$this->_version.$this->_resourceUri. | |
'?'.http_build_query($params); | |
// open connection | |
$connection = curl_init($fullUrl); | |
curl_setopt($connection, CURLOPT_RETURNTRANSFER, 1); | |
curl_setopt($connection, CURLOPT_USERAGENT, self::USER_AGENT); | |
curl_setopt($connection, CURLOPT_HEADER, 1); // return HTTP headers with response | |
if (null !== $postData) { | |
curl_setopt($connection, CURLOPT_HTTPHEADER, ['Content-Type: application/json']); | |
curl_setopt($connection, CURLOPT_POSTFIELDS, json_encode($postData, JSON_PRETTY_PRINT | JSON_UNESCAPED_SLASHES)); | |
curl_setopt($connection, CURLOPT_POST, 1); | |
} | |
// execute request | |
$resp = curl_exec($connection); | |
[$response_headers, $this->response_json] = explode("\r\n\r\n", $resp, 2); | |
// $response_headers now has a string of the HTTP headers | |
// $response_json is the body of the HTTP response | |
$headers = []; | |
foreach (explode("\r\n", $response_headers) as $i => $line) { | |
if (0 === $i) { | |
$headers['http_code'] = $line; | |
} else { | |
[$key, $value] = explode(': ', $line); | |
$headers[$key] = $value; | |
} | |
} | |
$this->response_code = curl_getinfo($connection, CURLINFO_HTTP_CODE); | |
$this->response_obj = json_decode($this->response_json); | |
if ('403' === $this->response_code) { | |
throw new NoCreditException($this->response_obj->message); | |
} else { | |
if ('200' === $this->response_code) { | |
$this->_update_rate_limit($headers); | |
} | |
} | |
return $this->response_obj; | |
} | |
} | |