summaryrefslogtreecommitdiffstats
path: root/lib/Service/CurlService.php
diff options
context:
space:
mode:
Diffstat (limited to 'lib/Service/CurlService.php')
-rw-r--r--lib/Service/CurlService.php315
1 files changed, 54 insertions, 261 deletions
diff --git a/lib/Service/CurlService.php b/lib/Service/CurlService.php
index 47d2f068..018b5de5 100644
--- a/lib/Service/CurlService.php
+++ b/lib/Service/CurlService.php
@@ -32,9 +32,15 @@ namespace OCA\Social\Service;
use daita\MySmallPhpTools\Exceptions\ArrayNotFoundException;
use daita\MySmallPhpTools\Exceptions\MalformedArrayException;
+use daita\MySmallPhpTools\Exceptions\RequestContentException;
+use daita\MySmallPhpTools\Exceptions\RequestNetworkException;
+use daita\MySmallPhpTools\Exceptions\RequestResultNotJsonException;
+use daita\MySmallPhpTools\Exceptions\RequestResultSizeException;
+use daita\MySmallPhpTools\Exceptions\RequestServerException;
use daita\MySmallPhpTools\Model\Request;
use daita\MySmallPhpTools\Traits\TArrayTools;
use daita\MySmallPhpTools\Traits\TPathTools;
+use daita\MySmallPhpTools\Traits\TRequest;
use Exception;
use OCA\Social\AP;
use OCA\Social\Exceptions\HostMetaException;
@@ -42,11 +48,6 @@ use OCA\Social\Exceptions\InvalidOriginException;
use OCA\Social\Exceptions\InvalidResourceException;
use OCA\Social\Exceptions\ItemUnknownException;
use OCA\Social\Exceptions\RedundancyLimitException;
-use OCA\Social\Exceptions\RequestContentException;
-use OCA\Social\Exceptions\RequestNetworkException;
-use OCA\Social\Exceptions\RequestResultNotJsonException;
-use OCA\Social\Exceptions\RequestResultSizeException;
-use OCA\Social\Exceptions\RequestServerException;
use OCA\Social\Exceptions\RetrieveAccountFormatException;
use OCA\Social\Exceptions\SocialAppConfigException;
use OCA\Social\Exceptions\UnauthorizedFediverseException;
@@ -57,6 +58,12 @@ class CurlService {
use TArrayTools;
use TPathTools;
+ use TRequest {
+ initRequestPost as initRequestPostOrig;
+ initRequestGet as initRequestGetOrig;
+ retrieveJson as retrieveJsonOrig;
+ doRequest as doRequestOrig;
+ }
const ASYNC_REQUEST_TOKEN = '/async/request/{token}';
@@ -73,13 +80,6 @@ class CurlService {
private $miscService;
- /** @var int */
- private $maxDownloadSize = 0;
-
- /** @var bool */
- private $maxDownloadSizeReached = false;
-
-
/**
* CurlService constructor.
*
@@ -93,6 +93,9 @@ class CurlService {
$this->configService = $configService;
$this->fediverseService = $fediverseService;
$this->miscService = $miscService;
+
+ $maxDlSize = $this->configService->getAppValue(ConfigService::SOCIAL_MAX_SIZE) * (1024 * 1024);
+ $this->setMaxDownloadSize($maxDlSize);
}
@@ -131,15 +134,8 @@ class CurlService {
$request = new Request($path);
$request->addData('resource', 'acct:' . $account);
$request->setAddress($host);
-
- try {
- $result = $this->request($request);
- } catch (RequestNetworkException $e) {
- if ($e->getCode() === CURLE_COULDNT_CONNECT) {
- $request->setProtocol('http');
- $result = $this->request($request);
- } else throw $e;
- }
+ $request->setProtocols(['https', 'http']);
+ $result = $this->retrieveJson($request);
$subject = $this->get('subject', $result, '');
list($type, $temp) = explode(':', $subject, 2);
@@ -162,7 +158,7 @@ class CurlService {
$request->setAddress($host);
try {
- $result = $this->request($request);
+ $result = $this->retrieveJson($request);
} catch (Exception $e) {
$this->miscService->log(
'hostMeta Exception - ' . get_class($e) . ' - ' . $e->getMessage(), 0
@@ -237,14 +233,13 @@ class CurlService {
* @throws UnauthorizedFediverseException
*/
public function retrieveObject($id): array {
-
$url = parse_url($id);
$this->mustContains(['path', 'host', 'scheme'], $url);
$request = new Request($url['path'], Request::TYPE_GET);
$request->setAddress($url['host']);
$request->setProtocol($url['scheme']);
- $result = $this->request($request);
+ $result = $this->retrieveJson($request);
if (is_array($result)) {
$result['_host'] = $request->getAddress();
}
@@ -256,60 +251,46 @@ class CurlService {
/**
* @param Request $request
*
- * @return mixed
+ * @return array
+ * @throws SocialAppConfigException
+ * @throws UnauthorizedFediverseException
* @throws RequestContentException
* @throws RequestNetworkException
* @throws RequestResultNotJsonException
* @throws RequestResultSizeException
* @throws RequestServerException
- * @throws SocialAppConfigException
- * @throws UnauthorizedFediverseException
*/
- public function request(Request $request) {
- $this->fediverseService->authorized($request->getAddress());
-
- $this->maxDownloadSizeReached = false;
- $this->assignUserAgent($request);
-
- $curl = $this->initRequest($request);
-
- $this->initRequestGet($request);
- $this->initRequestPost($curl, $request);
- $this->initRequestPut($curl, $request);
- $this->initRequestDelete($curl, $request);
-
- $this->initRequestHeaders($curl, $request);
-
- $result = curl_exec($curl);
-
- if ($this->maxDownloadSizeReached === true) {
- throw new RequestResultSizeException();
- }
- $this->parseRequestResult($curl, $request);
-
- if ($request->isBinary()) {
- $this->miscService->log('[>>] request (binary): ' . json_encode($request), 1);
-
- return $result;
+ public function retrieveJson(Request $request): array {
+ try {
+ $result = $this->retrieveJsonOrig($request);
+ } catch (RequestResultSizeException | RequestResultNotJsonException $e) {
+ $this->miscService->log(
+ '[!!] request: ' . json_encode($request) . ' - content-type: '
+ . $request->getContentType() . ' - ' . $e->getMessage(), 1
+ );
+ throw $e;
}
- $this->miscService->log(
- '[>>] request: ' . json_encode($request) . ' - content-type: '
- . $request->getContentType() . ' - result: ' . $result, 1
- );
+ return $result;
+ }
- if (strpos($request->getContentType(), 'application/xrd') === 0) {
- $xml = simplexml_load_string($result);
- $result = json_encode($xml, JSON_UNESCAPED_SLASHES);
- $this->miscService->log('XRD conversion to JSON: ' . $result, 1);
- }
- $result = json_decode((string)$result, true);
- if (is_array($result)) {
- return $result;
- }
+ /**
+ * @param Request $request
+ *
+ * @return mixed
+ * @throws SocialAppConfigException
+ * @throws UnauthorizedFediverseException
+ * @throws RequestContentException
+ * @throws RequestNetworkException
+ * @throws RequestResultSizeException
+ * @throws RequestServerException
+ */
+ public function doRequest(Request $request) {
+ $this->fediverseService->authorized($request->getAddress());
+ $this->assignUserAgent($request);
- throw new RequestResultNotJsonException();
+ return $this->doRequestOrig($request);
}
@@ -340,7 +321,7 @@ class CurlService {
$request->setProtocol(parse_url($address, PHP_URL_SCHEME));
try {
- $this->request($request);
+ $this->retrieveJson($request);
} catch (RequestResultNotJsonException $e) {
} catch (Exception $e) {
$this->miscService->log(
@@ -353,73 +334,8 @@ class CurlService {
/**
* @param Request $request
- *
- * @return resource
*/
- private function initRequest(Request $request) {
-
- $curl = $this->generateCurlRequest($request);
-
- curl_setopt($curl, CURLOPT_USERAGENT, $request->getUserAgent());
- curl_setopt($curl, CURLOPT_CONNECTTIMEOUT, $request->getTimeout());
- curl_setopt($curl, CURLOPT_TIMEOUT, $request->getTimeout());
-
- curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
- curl_setopt($curl, CURLOPT_BINARYTRANSFER, $request->isBinary());
-
- curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
- curl_setopt($curl, CURLOPT_FOLLOWLOCATION, true);
-
- $this->maxDownloadSize =
- $this->configService->getAppValue(ConfigService::SOCIAL_MAX_SIZE) * (1024 * 1024);
- curl_setopt($curl, CURLOPT_BUFFERSIZE, 128);
- curl_setopt($curl, CURLOPT_NOPROGRESS, false);
- curl_setopt(
- /**
- * @param $downloadSize
- * @param int $downloaded
- * @param $uploadSize
- * @param int $uploaded
- *
- * @return int
- */
- $curl, CURLOPT_PROGRESSFUNCTION,
- function($downloadSize, int $downloaded, $uploadSize, int $uploaded) {
- if ($downloaded > $this->maxDownloadSize) {
- $this->maxDownloadSizeReached = true;
-
- return 1;
- }
-
- return 0;
- }
- );
-
- return $curl;
- }
-
-
- /**
- * @param Request $request
- *
- * @return resource
- */
- private function generateCurlRequest(Request $request) {
- $url = $request->getProtocol() . '://' . $request->getAddress() . $request->getParsedUrl();
- if ($request->getType() !== Request::TYPE_GET) {
- $curl = curl_init($url);
- } else {
- $curl = curl_init($url . '?' . $request->getUrlData());
- }
-
- return $curl;
- }
-
-
- /**
- * @param Request $request
- */
- private function initRequestGet(Request $request) {
+ protected function initRequestGet(Request $request) {
if ($request->getType() !== Request::TYPE_GET) {
return;
}
@@ -430,6 +346,8 @@ class CurlService {
$request->addHeader(
'Accept: application/ld+json; profile="https://www.w3.org/ns/activitystreams"'
);
+
+ $this->initRequestGetOrig($request);
}
@@ -437,7 +355,7 @@ class CurlService {
* @param resource $curl
* @param Request $request
*/
- private function initRequestPost($curl, Request $request) {
+ protected function initRequestPost($curl, Request $request) {
if ($request->getType() !== Request::TYPE_POST) {
return;
}
@@ -446,132 +364,7 @@ class CurlService {
'Content-Type: application/ld+json; profile="https://www.w3.org/ns/activitystreams"'
);
- curl_setopt($curl, CURLOPT_POST, true);
- curl_setopt($curl, CURLOPT_POSTFIELDS, $request->getDataBody());
- }
-
-
- /**
- * @param resource $curl
- * @param Request $request
- */
- private function initRequestPut($curl, Request $request) {
- if ($request->getType() !== Request::TYPE_PUT) {
- return;
- }
-
- curl_setopt($curl, CURLOPT_CUSTOMREQUEST, "PUT");
- curl_setopt($curl, CURLOPT_POSTFIELDS, $request->getDataBody());
- }
-
-
- /**
- * @param resource $curl
- * @param Request $request
- */
- private function initRequestDelete($curl, Request $request) {
- if ($request->getType() !== Request::TYPE_DELETE) {
- return;
- }
-
- curl_setopt($curl, CURLOPT_CUSTOMREQUEST, "DELETE");
- curl_setopt($curl, CURLOPT_POSTFIELDS, $request->getDataBody());
- }
-
-
- /**
- * @param resource $curl
- * @param Request $request
- */
- private function initRequestHeaders($curl, Request $request) {
- $headers = $request->getHeaders();
-
- curl_setopt($curl, CURLOPT_HTTPHEADER, $headers);
- }
-
-
- /**
- * @param resource $curl
- * @param Request $request
- *
- * @throws RequestContentException
- * @throws RequestServerException
- * @throws RequestNetworkException
- */
- private function parseRequestResult($curl, Request &$request) {
- $this->parseRequestResultCurl($curl, $request);
-
- $code = curl_getinfo($curl, CURLINFO_HTTP_CODE);
- $type = curl_getinfo($curl, CURLINFO_CONTENT_TYPE);
- $request->setContentType((is_null($type) || is_bool($type)) ? '' : $type);
- $request->setResultCode($code);
-
- $this->parseRequestResultCode301($code, $request);
- $this->parseRequestResultCode4xx($code, $request);
- $this->parseRequestResultCode5xx($code, $request);
- }
-
-
- /**
- * @param resource $curl
- * @param Request $request
- *
- * @throws RequestNetworkException
- */
- private function parseRequestResultCurl($curl, Request $request) {
- $errno = curl_errno($curl);
- if ($errno > 0) {
- throw new RequestNetworkException(
- $errno . ' - ' . curl_error($curl) . ' - ' . json_encode(
- $request, JSON_UNESCAPED_SLASHES
- ), $errno
- );
- }
- }
-
-
- /**
- * @param int $code
- * @param Request $request
- *
- * @throws RequestContentException
- */
- private function parseRequestResultCode301($code, Request $request) {
- if ($code === 301) {
- throw new RequestContentException(
- '301 - ' . json_encode($request, JSON_UNESCAPED_SLASHES)
- );
- }
- }
-
-
- /**
- * @param int $code
- * @param Request $request
- *
- * @throws RequestContentException
- */
- private function parseRequestResultCode4xx(int $code, Request $request) {
- if ($code === 404 || $code === 410) {
- throw new RequestContentException(
- $code . ' - ' . json_encode($request, JSON_UNESCAPED_SLASHES)
- );
- }
- }
-
-
- /**
- * @param int $code
- * @param Request $request
- *
- * @throws RequestServerException
- */
- private function parseRequestResultCode5xx(int $code, Request $request) {
- if ($code === 500) {
- throw new RequestServerException(
- $code . ' - ' . json_encode($request, JSON_UNESCAPED_SLASHES)
- );
- }
+ $this->initRequestPostOrig($curl, $request);
}
}