kopia lustrzana https://github.com/nextcloud/social
				
				
				
			
		
			
				
	
	
		
			292 wiersze
		
	
	
		
			7.4 KiB
		
	
	
	
		
			PHP
		
	
	
			
		
		
	
	
			292 wiersze
		
	
	
		
			7.4 KiB
		
	
	
	
		
			PHP
		
	
	
<?php
 | 
						|
 | 
						|
declare(strict_types=1);
 | 
						|
 | 
						|
/**
 | 
						|
 * SPDX-FileCopyrightText: 2018 Nextcloud GmbH and Nextcloud contributors
 | 
						|
 * SPDX-License-Identifier: AGPL-3.0-or-later
 | 
						|
 */
 | 
						|
 | 
						|
namespace OCA\Social\Controller;
 | 
						|
 | 
						|
use Exception;
 | 
						|
use OCA\Social\AppInfo\Application;
 | 
						|
use OCA\Social\Exceptions\ClientException;
 | 
						|
use OCA\Social\Exceptions\ClientNotFoundException;
 | 
						|
use OCA\Social\Exceptions\InstanceDoesNotExistException;
 | 
						|
use OCA\Social\Model\Client\SocialClient;
 | 
						|
use OCA\Social\Service\AccountService;
 | 
						|
use OCA\Social\Service\ClientService;
 | 
						|
use OCA\Social\Service\ConfigService;
 | 
						|
use OCA\Social\Service\InstanceService;
 | 
						|
use OCP\AppFramework\Controller;
 | 
						|
use OCP\AppFramework\Http;
 | 
						|
use OCP\AppFramework\Http\DataResponse;
 | 
						|
use OCP\AppFramework\Http\Response;
 | 
						|
use OCP\AppFramework\Http\TemplateResponse;
 | 
						|
use OCP\AppFramework\Services\IInitialState;
 | 
						|
use OCP\IRequest;
 | 
						|
use OCP\IURLGenerator;
 | 
						|
use OCP\IUserSession;
 | 
						|
use Psr\Log\LoggerInterface;
 | 
						|
 | 
						|
class OAuthController extends Controller {
 | 
						|
	private IUserSession $userSession;
 | 
						|
	private IURLGenerator $urlGenerator;
 | 
						|
	private InstanceService $instanceService;
 | 
						|
	private AccountService $accountService;
 | 
						|
	private ClientService $clientService;
 | 
						|
	private ConfigService $configService;
 | 
						|
	private LoggerInterface $logger;
 | 
						|
	private IInitialState $initialState;
 | 
						|
 | 
						|
	public function __construct(
 | 
						|
		IRequest $request,
 | 
						|
		IUserSession $userSession,
 | 
						|
		IURLGenerator $urlGenerator,
 | 
						|
		InstanceService $instanceService,
 | 
						|
		AccountService $accountService,
 | 
						|
		ClientService $clientService,
 | 
						|
		ConfigService $configService,
 | 
						|
		LoggerInterface $logger,
 | 
						|
		IInitialState $initialState,
 | 
						|
	) {
 | 
						|
		parent::__construct(Application::APP_ID, $request);
 | 
						|
 | 
						|
		$this->userSession = $userSession;
 | 
						|
		$this->urlGenerator = $urlGenerator;
 | 
						|
		$this->instanceService = $instanceService;
 | 
						|
		$this->accountService = $accountService;
 | 
						|
		$this->clientService = $clientService;
 | 
						|
		$this->configService = $configService;
 | 
						|
		$this->logger = $logger;
 | 
						|
		$this->initialState = $initialState;
 | 
						|
 | 
						|
		$body = file_get_contents('php://input');
 | 
						|
		$logger->debug('[OAuthController] input: ' . $body);
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * @NoCSRFRequired
 | 
						|
	 * @PublicPage
 | 
						|
	 */
 | 
						|
	public function nodeinfo2(): Response {
 | 
						|
		try {
 | 
						|
			$local = $this->instanceService->getLocal();
 | 
						|
			$name = $local->getTitle();
 | 
						|
 | 
						|
			$version = $local->getVersion();
 | 
						|
			$usage = $local->getUsage();
 | 
						|
			$openReg = $local->isRegistrations();
 | 
						|
		} catch (InstanceDoesNotExistException $e) {
 | 
						|
			$name = 'Nextcloud Social';
 | 
						|
			$version = $this->configService->getAppValue('installed_version');
 | 
						|
			$usage = [];
 | 
						|
			$openReg = false;
 | 
						|
		}
 | 
						|
 | 
						|
		$nodeInfo = [
 | 
						|
			'version' => '2.0',
 | 
						|
			'software' => [
 | 
						|
				'name' => $name,
 | 
						|
				'version' => $version
 | 
						|
			],
 | 
						|
			'protocols' => [
 | 
						|
				'activitypub'
 | 
						|
			],
 | 
						|
			'rootUrl' => rtrim($this->urlGenerator->linkToRouteAbsolute('social.Navigation.navigate'), '/'),
 | 
						|
			'usage' => $usage,
 | 
						|
			'openRegistrations' => $openReg
 | 
						|
		];
 | 
						|
 | 
						|
		return new DataResponse($nodeInfo, Http::STATUS_OK);
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * @NoCSRFRequired
 | 
						|
	 * @PublicPage
 | 
						|
	 *
 | 
						|
	 * @param array|string $redirect_uris
 | 
						|
	 *
 | 
						|
	 * @throws ClientException
 | 
						|
	 */
 | 
						|
	public function apps(
 | 
						|
		string $client_name = '',
 | 
						|
		$redirect_uris = '',
 | 
						|
		string $website = '',
 | 
						|
		string $scopes = 'read',
 | 
						|
	): DataResponse {
 | 
						|
		// TODO: manage array from request
 | 
						|
		if (!is_array($redirect_uris)) {
 | 
						|
			$redirect_uris = [$redirect_uris];
 | 
						|
		}
 | 
						|
 | 
						|
		$client = new SocialClient();
 | 
						|
		$client->setAppWebsite($website);
 | 
						|
		$client->setAppRedirectUris($redirect_uris);
 | 
						|
		$client->setAppScopes($client->getScopesFromString($scopes));
 | 
						|
		$client->setAppName($client_name);
 | 
						|
 | 
						|
		$this->clientService->createApp($client);
 | 
						|
 | 
						|
		return new DataResponse(
 | 
						|
			[
 | 
						|
				'id' => $client->getId(),
 | 
						|
				'name' => $client->getAppName(),
 | 
						|
				'website' => $client->getAppWebsite(),
 | 
						|
				'scopes' => implode(' ', $client->getAppScopes()),
 | 
						|
				'client_id' => $client->getAppClientId(),
 | 
						|
				'client_secret' => $client->getAppClientSecret()
 | 
						|
			], Http::STATUS_OK
 | 
						|
		);
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * @NoCSRFRequired
 | 
						|
	 * @NoAdminRequired
 | 
						|
	 */
 | 
						|
	public function authorize(
 | 
						|
		string $client_id,
 | 
						|
		string $redirect_uri,
 | 
						|
		string $response_type,
 | 
						|
		string $scope = 'read',
 | 
						|
	): Response {
 | 
						|
		$user = $this->userSession->getUser();
 | 
						|
 | 
						|
		// check actor exists
 | 
						|
		$this->accountService->getActorFromUserId($user->getUID());
 | 
						|
 | 
						|
		if ($response_type !== 'code') {
 | 
						|
			throw new ClientNotFoundException('invalid response type');
 | 
						|
		}
 | 
						|
 | 
						|
		// check client exists in db
 | 
						|
		$client = $this->clientService->getFromClientId($client_id);
 | 
						|
		$this->initialState->provideInitialState('appName', $client->getAppName());
 | 
						|
 | 
						|
		return new TemplateResponse(Application::APP_ID, 'oauth2', [
 | 
						|
			'request' =>
 | 
						|
				[
 | 
						|
					'clientId' => $client_id,
 | 
						|
					'redirectUri' => $redirect_uri,
 | 
						|
					'responseType' => $response_type,
 | 
						|
					'scope' => $scope
 | 
						|
				]
 | 
						|
		]);
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * @NoAdminRequired
 | 
						|
	 */
 | 
						|
	public function authorizing(
 | 
						|
		string $client_id,
 | 
						|
		string $redirect_uri,
 | 
						|
		string $response_type,
 | 
						|
		string $scope = 'read',
 | 
						|
	): DataResponse {
 | 
						|
		try {
 | 
						|
			$user = $this->userSession->getUser();
 | 
						|
			$account = $this->accountService->getActorFromUserId($user->getUID());
 | 
						|
 | 
						|
			if ($response_type !== 'code') {
 | 
						|
				throw new ClientNotFoundException('invalid response type');
 | 
						|
			}
 | 
						|
 | 
						|
			$client = $this->clientService->getFromClientId($client_id);
 | 
						|
			$this->clientService->confirmData(
 | 
						|
				$client,
 | 
						|
				[
 | 
						|
					'app_scopes' => $scope,
 | 
						|
					'redirect_uri', $redirect_uri
 | 
						|
				]
 | 
						|
			);
 | 
						|
 | 
						|
			$client->setAuthScopes($client->getScopesFromString($scope));
 | 
						|
			$client->setAuthAccount($account->getPreferredUsername());
 | 
						|
			$client->setAuthUserId($user->getUID());
 | 
						|
 | 
						|
			$this->clientService->authClient($client);
 | 
						|
			$code = $client->getAuthCode();
 | 
						|
 | 
						|
			if ($redirect_uri !== 'urn:ietf:wg:oauth:2.0:oob') {
 | 
						|
				header('Location: ' . $redirect_uri . '?code=' . $code);
 | 
						|
				exit();
 | 
						|
			}
 | 
						|
 | 
						|
			// TODO : finalize result if no redirect_url
 | 
						|
			return new DataResponse(
 | 
						|
				['code' => $code], Http::STATUS_OK
 | 
						|
			);
 | 
						|
		} catch (Exception $e) {
 | 
						|
			$this->logger->notice($e->getMessage() . ' ' . get_class($e));
 | 
						|
 | 
						|
			return new DataResponse(['error' => $e->getMessage()], Http::STATUS_BAD_REQUEST);
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * @NoCSRFRequired
 | 
						|
	 * @NoAdminRequired
 | 
						|
	 * @PublicPage
 | 
						|
	 */
 | 
						|
	public function token(
 | 
						|
		string $client_id,
 | 
						|
		string $client_secret,
 | 
						|
		string $redirect_uri,
 | 
						|
		string $grant_type,
 | 
						|
		string $scope = 'read',
 | 
						|
		string $code = '',
 | 
						|
	): DataResponse {
 | 
						|
		try {
 | 
						|
			$client = $this->clientService->getFromClientId($client_id);
 | 
						|
			$this->clientService->confirmData(
 | 
						|
				$client,
 | 
						|
				[
 | 
						|
					'client_secret' => $client_secret,
 | 
						|
					'redirect_uri' => $redirect_uri,
 | 
						|
					'auth_scopes' => $scope
 | 
						|
				]
 | 
						|
			);
 | 
						|
 | 
						|
			if ($grant_type === 'authorization_code') {
 | 
						|
				if ($code === '') {
 | 
						|
					return new DataResponse(['error' => 'missing code'], Http::STATUS_BAD_REQUEST);
 | 
						|
				}
 | 
						|
 | 
						|
				$this->clientService->confirmData($client, ['code' => $code]);
 | 
						|
				$this->clientService->generateToken($client);
 | 
						|
			} elseif ($grant_type === 'client_credentials') {
 | 
						|
				// TODO: manage client_credentials
 | 
						|
			} else {
 | 
						|
				return new DataResponse(
 | 
						|
					['error' => 'invalid value for grant_type'], Http::STATUS_BAD_REQUEST
 | 
						|
				);
 | 
						|
			}
 | 
						|
 | 
						|
			if ($client->getToken() === '') {
 | 
						|
				return new DataResponse(
 | 
						|
					['error' => 'issue generating access_token'], Http::STATUS_BAD_REQUEST
 | 
						|
				);
 | 
						|
			}
 | 
						|
 | 
						|
			return new DataResponse(
 | 
						|
				[
 | 
						|
					'access_token' => $client->getToken(),
 | 
						|
					'token_type' => 'Bearer',
 | 
						|
					'scope' => $scope,
 | 
						|
					'created_at' => $client->getCreation()
 | 
						|
				], Http::STATUS_OK
 | 
						|
			);
 | 
						|
		} catch (ClientNotFoundException $e) {
 | 
						|
			return new DataResponse(['error' => 'unknown client_id'], Http::STATUS_UNAUTHORIZED);
 | 
						|
		} catch (Exception $e) {
 | 
						|
			return new DataResponse(['error' => $e->getMessage()], Http::STATUS_UNAUTHORIZED);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 |