kopia lustrzana https://github.com/nextcloud/social
				
				
				
			
		
			
				
	
	
		
			551 wiersze
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
			
		
		
	
	
			551 wiersze
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
| <?php
 | |
| declare(strict_types=1);
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Nextcloud - Social Support
 | |
|  *
 | |
|  * This file is licensed under the Affero General Public License version 3 or
 | |
|  * later. See the COPYING file.
 | |
|  *
 | |
|  * @author Maxence Lange <maxence@artificial-owl.com>
 | |
|  * @copyright 2018, Maxence Lange <maxence@artificial-owl.com>
 | |
|  * @license GNU AGPL version 3 or any later version
 | |
|  *
 | |
|  * 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 <http://www.gnu.org/licenses/>.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| namespace OCA\Social\Service;
 | |
| 
 | |
| 
 | |
| use daita\MySmallPhpTools\Exceptions\ArrayNotFoundException;
 | |
| use daita\MySmallPhpTools\Exceptions\MalformedArrayException;
 | |
| use daita\MySmallPhpTools\Model\Request;
 | |
| use daita\MySmallPhpTools\Traits\TArrayTools;
 | |
| use daita\MySmallPhpTools\Traits\TPathTools;
 | |
| use Exception;
 | |
| use OCA\Social\AP;
 | |
| use OCA\Social\Exceptions\HostMetaException;
 | |
| use OCA\Social\Exceptions\InvalidOriginException;
 | |
| use OCA\Social\Exceptions\InvalidResourceException;
 | |
| use OCA\Social\Exceptions\RedundancyLimitException;
 | |
| use OCA\Social\Exceptions\RequestContentException;
 | |
| use OCA\Social\Exceptions\RequestResultNotJsonException;
 | |
| use OCA\Social\Exceptions\RetrieveAccountFormatException;
 | |
| use OCA\Social\Exceptions\RequestNetworkException;
 | |
| use OCA\Social\Exceptions\RequestResultSizeException;
 | |
| use OCA\Social\Exceptions\RequestServerException;
 | |
| use OCA\Social\Exceptions\SocialAppConfigException;
 | |
| use OCA\Social\Exceptions\ItemUnknownException;
 | |
| use OCA\Social\Exceptions\UnauthorizedFediverseException;
 | |
| use OCA\Social\Model\ActivityPub\Actor\Person;
 | |
| use OCA\Social\Model\ActivityPub\Actor\Service;
 | |
| 
 | |
| class CurlService {
 | |
| 
 | |
| 
 | |
| 	use TArrayTools;
 | |
| 	use TPathTools;
 | |
| 
 | |
| 
 | |
| 	const ASYNC_REQUEST_TOKEN = '/async/request/{token}';
 | |
| 	const USER_AGENT = 'Nextcloud Social';
 | |
| 
 | |
| 
 | |
| 	/** @var ConfigService */
 | |
| 	private $configService;
 | |
| 
 | |
| 	/** @var FediverseService */
 | |
| 	private $fediverseService;
 | |
| 
 | |
| 	/** @var MiscService */
 | |
| 	private $miscService;
 | |
| 
 | |
| 
 | |
| 	/** @var int */
 | |
| 	private $maxDownloadSize = 0;
 | |
| 
 | |
| 	/** @var bool */
 | |
| 	private $maxDownloadSizeReached = false;
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * CurlService constructor.
 | |
| 	 *
 | |
| 	 * @param ConfigService $configService
 | |
| 	 * @param FediverseService $fediverseService
 | |
| 	 * @param MiscService $miscService
 | |
| 	 */
 | |
| 	public function __construct(
 | |
| 		ConfigService $configService, FediverseService $fediverseService, MiscService $miscService
 | |
| 	) {
 | |
| 		$this->configService = $configService;
 | |
| 		$this->fediverseService = $fediverseService;
 | |
| 		$this->miscService = $miscService;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @param string $account
 | |
| 	 *
 | |
| 	 * @return array
 | |
| 	 * @throws InvalidResourceException
 | |
| 	 * @throws RequestContentException
 | |
| 	 * @throws RequestNetworkException
 | |
| 	 * @throws RequestResultNotJsonException
 | |
| 	 * @throws RequestResultSizeException
 | |
| 	 * @throws RequestServerException
 | |
| 	 * @throws SocialAppConfigException
 | |
| 	 * @throws UnauthorizedFediverseException
 | |
| 	 */
 | |
| 	public function webfingerAccount(string &$account): array {
 | |
| 		$account = $this->withoutBeginAt($account);
 | |
| 
 | |
| 		// we consider an account is like an email
 | |
| 		if (!filter_var($account, FILTER_VALIDATE_EMAIL)) {
 | |
| 			throw new InvalidResourceException('account format is not valid');
 | |
| 		}
 | |
| 
 | |
| 		list($username, $host) = explode('@', $account);
 | |
| 		if ($username === null || $host === null) {
 | |
| 			throw new InvalidResourceException();
 | |
| 		}
 | |
| 
 | |
| 		try {
 | |
| 			$path = $this->hostMeta($host);
 | |
| 		} catch (HostMetaException $e) {
 | |
| 			$path = '/.well-known/webfinger';
 | |
| 		}
 | |
| 
 | |
| 		$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;
 | |
| 		}
 | |
| 
 | |
| 		$subject = $this->get('subject', $result, '');
 | |
| 		list($type, $temp) = explode(':', $subject, 2);
 | |
| 		if ($type === 'acct') {
 | |
| 			$account = $temp;
 | |
| 		}
 | |
| 
 | |
| 		return $result;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @param string $host
 | |
| 	 *
 | |
| 	 * @return string
 | |
| 	 * @throws HostMetaException
 | |
| 	 */
 | |
| 	public function hostMeta(string &$host): string {
 | |
| 		$request = new Request('/.well-known/host-meta');
 | |
| 		$request->setAddress($host);
 | |
| 
 | |
| 		try {
 | |
| 			$result = $this->request($request);
 | |
| 		} catch (Exception $e) {
 | |
| 			$this->miscService->log(
 | |
| 				'hostMeta Exception - ' . get_class($e) . ' - ' . $e->getMessage(), 0
 | |
| 			);
 | |
| 			throw new HostMetaException($e->getMessage());
 | |
| 		}
 | |
| 
 | |
| 		$url = $this->get('Link.@attributes.template', $result, '');
 | |
| 		$host = parse_url($url, PHP_URL_HOST);
 | |
| 
 | |
| 		return parse_url($url, PHP_URL_PATH);
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @param string $account
 | |
| 	 *
 | |
| 	 * @return Person
 | |
| 	 * @throws InvalidOriginException
 | |
| 	 * @throws InvalidResourceException
 | |
| 	 * @throws MalformedArrayException
 | |
| 	 * @throws RedundancyLimitException
 | |
| 	 * @throws RequestContentException
 | |
| 	 * @throws RetrieveAccountFormatException
 | |
| 	 * @throws RequestNetworkException
 | |
| 	 * @throws RequestResultSizeException
 | |
| 	 * @throws RequestServerException
 | |
| 	 * @throws SocialAppConfigException
 | |
| 	 * @throws ItemUnknownException
 | |
| 	 * @throws RequestResultNotJsonException
 | |
| 	 * @throws UnauthorizedFediverseException
 | |
| 	 */
 | |
| 	public function retrieveAccount(string &$account): Person {
 | |
| 		$result = $this->webfingerAccount($account);
 | |
| 
 | |
| 		try {
 | |
| 			$link = $this->extractArray('rel', 'self', $this->getArray('links', $result));
 | |
| 		} catch (ArrayNotFoundException $e) {
 | |
| 			throw new RetrieveAccountFormatException();
 | |
| 		}
 | |
| 
 | |
| 		$id = $this->get('href', $link, '');
 | |
| 		$data = $this->retrieveObject($id);
 | |
| 
 | |
| 		/** @var Person $actor */
 | |
| 		$actor = AP::$activityPub->getItemFromData($data);
 | |
| 		if (!AP::$activityPub->isActor($actor)) {
 | |
| 			throw new ItemUnknownException();
 | |
| 		}
 | |
| 
 | |
| 		if (strtolower($actor->getId()) !== strtolower($id)) {
 | |
| 			throw new InvalidOriginException(
 | |
| 				'CurlService::retrieveAccount - id: ' . $id . ' - actorId: ' . $actor->getId()
 | |
| 			);
 | |
| 		}
 | |
| 
 | |
| 		return $actor;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $id
 | |
| 	 *
 | |
| 	 * @return array
 | |
| 	 * @throws MalformedArrayException
 | |
| 	 * @throws RequestContentException
 | |
| 	 * @throws RequestNetworkException
 | |
| 	 * @throws RequestResultNotJsonException
 | |
| 	 * @throws RequestResultSizeException
 | |
| 	 * @throws RequestServerException
 | |
| 	 * @throws SocialAppConfigException
 | |
| 	 * @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);
 | |
| 		if (is_array($result)) {
 | |
| 			$result['_host'] = $request->getAddress();
 | |
| 		}
 | |
| 
 | |
| 		return $result;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @param Request $request
 | |
| 	 *
 | |
| 	 * @return mixed
 | |
| 	 * @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->initRequestPost($curl, $request);
 | |
| 		$this->initRequestPut($curl, $request);
 | |
| 		$this->initRequestDelete($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;
 | |
| 		}
 | |
| 
 | |
| 		$this->miscService->log(
 | |
| 			'[>>] request: ' . json_encode($request) . ' - content-type: '
 | |
| 			. $request->getContentType() . ' - result: ' . $result, 1
 | |
| 		);
 | |
| 
 | |
| 		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;
 | |
| 		}
 | |
| 
 | |
| 		throw new RequestResultNotJsonException();
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @param Request $request
 | |
| 	 */
 | |
| 	public function assignUserAgent(Request $request) {
 | |
| 		$request->setUserAgent(
 | |
| 			self::USER_AGENT . ' ' . $this->configService->getAppValue('installed_version')
 | |
| 		);
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @param string $token
 | |
| 	 *
 | |
| 	 * @throws SocialAppConfigException
 | |
| 	 */
 | |
| 	public function asyncWithToken(string $token) {
 | |
| 		$address = $this->configService->getUrlSocial();
 | |
| 		$parse = parse_url($address);
 | |
| 		$host = $this->get('host', $parse, '');
 | |
| 		$path = $this->withEndSlash($this->get('path', $parse, '')) . $this->withoutBeginSlash(
 | |
| 				self::ASYNC_REQUEST_TOKEN
 | |
| 			);
 | |
| 		$path = str_replace('{token}', $token, $path);
 | |
| 
 | |
| 		$request = new Request($path, Request::TYPE_POST);
 | |
| 		$request->setAddress($host);
 | |
| 		$request->setProtocol($this->get('scheme', $parse, 'https'));
 | |
| 
 | |
| 		try {
 | |
| 			$this->request($request);
 | |
| 		} catch (RequestResultNotJsonException $e) {
 | |
| 		} catch (Exception $e) {
 | |
| 			$this->miscService->log(
 | |
| 				'Cannot initiate AsyncWithToken ' . json_encode($token) . ' (' . get_class($e)
 | |
| 				. ' - ' . json_encode($e) . ')', 1
 | |
| 			);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @param Request $request
 | |
| 	 *
 | |
| 	 * @return resource
 | |
| 	 */
 | |
| 	private function initRequest(Request $request) {
 | |
| 
 | |
| 		$curl = $this->generateCurlRequest($request);
 | |
| 		$headers = $request->getHeaders();
 | |
| 
 | |
| 		$headers[] = 'Accept: application/ld+json; profile="https://www.w3.org/ns/activitystreams"';
 | |
| 
 | |
| 		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_HTTPHEADER, $headers);
 | |
| 
 | |
| 		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 resource $curl
 | |
| 	 * @param Request $request
 | |
| 	 */
 | |
| 	private function initRequestPost($curl, Request $request) {
 | |
| 		if ($request->getType() !== Request::TYPE_POST) {
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		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
 | |
| 	 *
 | |
| 	 * @throws RequestContentException
 | |
| 	 * @throws RequestServerException
 | |
| 	 * @throws RequestNetworkException
 | |
| 	 */
 | |
| 	private function parseRequestResult($curl, Request &$request) {
 | |
| 		$this->parseRequestResultCurl($curl, $request);
 | |
| 
 | |
| 		$code = curl_getinfo($curl, CURLINFO_HTTP_CODE);
 | |
| 		$contentType = curl_getinfo($curl, CURLINFO_CONTENT_TYPE);
 | |
| 		$request->setContentType(($contentType === null) ? '' : $contentType);
 | |
| 		$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)
 | |
| 			);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| }
 | |
| 
 |