mirror of
https://github.com/espocrm/espocrm.git
synced 2026-06-28 06:56:05 +00:00
201 lines
6.2 KiB
PHP
201 lines
6.2 KiB
PHP
<?php
|
||
/************************************************************************
|
||
* This file is part of EspoCRM.
|
||
*
|
||
* EspoCRM – Open Source CRM application.
|
||
* Copyright (C) 2014-2026 EspoCRM, Inc.
|
||
* Website: https://www.espocrm.com
|
||
*
|
||
* This program is free software: you can redistribute it and/or modify
|
||
* it under the terms of the GNU Affero General Public License as published by
|
||
* the Free Software Foundation, either version 3 of the License, or
|
||
* (at your option) any later version.
|
||
*
|
||
* This program is distributed in the hope that it will be useful,
|
||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
* GNU Affero General Public License for more details.
|
||
*
|
||
* You should have received a copy of the GNU Affero General Public License
|
||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||
*
|
||
* The interactive user interfaces in modified source and object code versions
|
||
* of this program must display Appropriate Legal Notices, as required under
|
||
* Section 5 of the GNU Affero General Public License version 3.
|
||
*
|
||
* In accordance with Section 7(b) of the GNU Affero General Public License version 3,
|
||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||
************************************************************************/
|
||
|
||
namespace Espo\Core\Webhook;
|
||
|
||
use Espo\Core\Exceptions\Error;
|
||
use Espo\Core\Utils\Config;
|
||
use Espo\Core\Utils\Json;
|
||
use Espo\Core\Utils\Security\UrlCheck;
|
||
use Espo\Entities\Webhook;
|
||
use Espo\Core\HttpClient;
|
||
use LogicException;
|
||
use Psr\Http\Message\RequestInterface;
|
||
|
||
/**
|
||
* Sends a portion.
|
||
*/
|
||
class Sender
|
||
{
|
||
private const int CONNECT_TIMEOUT = 5;
|
||
private const int TIMEOUT = 10;
|
||
|
||
public function __construct(
|
||
private Config $config,
|
||
private UrlCheck $urlCheck,
|
||
private HttpClient\ClientFactory $clientFactory,
|
||
) {}
|
||
|
||
/**
|
||
* @param array<int, mixed> $dataList
|
||
* @throws Error
|
||
*/
|
||
public function send(Webhook $webhook, array $dataList): int
|
||
{
|
||
$payload = Json::encode($dataList);
|
||
|
||
[$signature, $legacySignature] = $this->prepareSignatures($webhook, $payload);
|
||
|
||
$options = new HttpClient\Options(
|
||
protocols: [HttpClient\Protocol::https, HttpClient\Protocol::http],
|
||
redirect: new HttpClient\Options\Redirect(
|
||
allow: true,
|
||
protocols: [HttpClient\Protocol::https],
|
||
),
|
||
timeout: $this->getTimeout(),
|
||
connectTimeout: $this->getConnectTimeout(),
|
||
internalHostRestriction: new HttpClient\Options\InternalHostRestriction(
|
||
restrict: true,
|
||
allowed: $this->getAllowedAddressList(),
|
||
),
|
||
);
|
||
|
||
$request = $this->prepareRequest(
|
||
url: $this->getUrl($webhook),
|
||
payload: $payload,
|
||
signature: $signature,
|
||
legacySignature: $legacySignature,
|
||
);
|
||
|
||
$client = $this->clientFactory->create($options);
|
||
|
||
try {
|
||
$response = $client->send($request);
|
||
} catch (HttpClient\Exceptions\ConnectException $e) {
|
||
if ($e->getReason() === HttpClient\ConnectErrorReason::Timeout) {
|
||
return 408;
|
||
}
|
||
|
||
throw new Error("Connect error.", previous: $e);
|
||
} catch (HttpClient\Exceptions\TooManyRedirectsException $e) {
|
||
throw new Error("Too many redirects.", previous: $e);
|
||
}
|
||
|
||
return $response->getStatusCode();
|
||
}
|
||
|
||
private function buildSignature(Webhook $webhook, string $payload, string $secretKey): string
|
||
{
|
||
$webhookId = $webhook->getId();
|
||
$hash = hash_hmac('sha256', $payload, $secretKey);
|
||
|
||
return base64_encode("$webhookId:$hash");
|
||
}
|
||
|
||
/**
|
||
* @todo Remove in v11.0.
|
||
*/
|
||
private function buildSignatureLegacy(Webhook $webhook, string $payload, string $secretKey): string
|
||
{
|
||
return base64_encode($webhook->getId() . ':' . hash_hmac('sha256', $payload, $secretKey, true));
|
||
}
|
||
|
||
/**
|
||
* @return string
|
||
* @throws Error
|
||
*/
|
||
private function getUrl(Webhook $webhook): string
|
||
{
|
||
$url = $webhook->getUrl() ?? throw new Error("Webhook does not have URL.");
|
||
|
||
if (!$this->urlCheck->isUrl($url)) {
|
||
throw new Error("'$url' is not valid URL.");
|
||
}
|
||
|
||
return $url;
|
||
}
|
||
|
||
/**
|
||
* @return string[]
|
||
*/
|
||
private function getAllowedAddressList(): array
|
||
{
|
||
/** @var string[] $allowedAddressList */
|
||
$allowedAddressList = $this->config->get('webhookAllowedAddressList') ?? [];
|
||
|
||
return $allowedAddressList;
|
||
}
|
||
|
||
private function prepareRequest(
|
||
string $url,
|
||
string $payload,
|
||
?string $signature,
|
||
?string $legacySignature,
|
||
): RequestInterface {
|
||
|
||
$request = HttpClient\RequestCreator::create('POST', $url)
|
||
->withHeader('Content-Type', 'application/json')
|
||
->withHeader('Content-Length', (string) strlen($payload));
|
||
|
||
if ($signature) {
|
||
$request = $request->withHeader('Signature', $signature);
|
||
}
|
||
|
||
if ($legacySignature) {
|
||
$request = $request->withHeader('X-Signature', $legacySignature);
|
||
}
|
||
|
||
$request = $request->withBody(HttpClient\Util::streamFor($payload));
|
||
|
||
if (!$request instanceof RequestInterface) {
|
||
throw new LogicException();
|
||
}
|
||
|
||
return $request;
|
||
}
|
||
|
||
/**
|
||
* @return array{0: ?string, 1: ?string}
|
||
*/
|
||
private function prepareSignatures(Webhook $webhook, string $payload): array
|
||
{
|
||
$signature = null;
|
||
$legacySignature = null;
|
||
|
||
$secretKey = $webhook->getSecretKey();
|
||
|
||
if ($secretKey) {
|
||
$signature = $this->buildSignature($webhook, $payload, $secretKey);
|
||
$legacySignature = $this->buildSignatureLegacy($webhook, $payload, $secretKey);
|
||
}
|
||
|
||
return [$signature, $legacySignature];
|
||
}
|
||
|
||
private function getConnectTimeout(): ?int
|
||
{
|
||
return $this->config->get('webhookConnectTimeout', self::CONNECT_TIMEOUT);
|
||
}
|
||
|
||
private function getTimeout(): ?int
|
||
{
|
||
return $this->config->get('webhookTimeout', self::TIMEOUT);
|
||
}
|
||
}
|