Files
Enrique Pérez Arnaud 2c54011789 feat(dav): accept bearer access tokens on webdav endpoints
Signed-off-by: Micke Nordin <kano@sunet.se>
Signed-off-by: Enrique Pérez Arnaud <enrique@cazalla.net>
2026-06-17 10:44:53 +02:00

96 lines
2.8 KiB
PHP

<?php
/**
* SPDX-FileCopyrightText: 2017 Nextcloud GmbH and Nextcloud contributors
* SPDX-License-Identifier: AGPL-3.0-or-later
*/
namespace OCA\DAV\Connector\Sabre;
use OCP\AppFramework\Http;
use OCP\Defaults;
use OCP\IConfig;
use OCP\IRequest;
use OCP\ISession;
use OCP\IUserSession;
use OCP\Server;
use OCP\Share\IManager;
use OCP\Share\IShare;
use Sabre\DAV\Auth\Backend\AbstractBearer;
use Sabre\HTTP\RequestInterface;
use Sabre\HTTP\ResponseInterface;
class BearerAuth extends AbstractBearer {
public function __construct(
private IUserSession $userSession,
private ISession $session,
private IRequest $request,
private IConfig $config,
private string $principalPrefix = 'principals/users/',
private string $token = '',
) {
// setup realm
$defaults = new Defaults();
$this->realm = $defaults->getName() ?: 'Nextcloud';
}
private function setupUserFs($userId) {
\OC_Util::setupFS($userId);
$this->session->close();
return $this->principalPrefix . $userId;
}
/**
* {@inheritdoc}
*/
#[\Override]
public function validateBearerToken($bearerToken) {
\OC_Util::setupFS();
$this->token = $bearerToken;
// public.php sets incognito mode for anonymous share access, which makes
// Session::getUser() return null and consequently Session::isLoggedIn()
// return false even after a successful token login. Disable it here so
// the logged-in user is visible for the rest of the request. If the
// bearer token is invalid and Sabre falls back to one of the public
// auth backends, that backend will re-enable incognito mode itself.
\OC_User::setIncognitoMode(false);
if (!$this->userSession->isLoggedIn()) {
$this->userSession->tryTokenLogin($this->request);
}
if ($this->userSession->isLoggedIn()) {
return $this->setupUserFs($this->userSession->getUser()->getUID());
}
return false;
}
public function getShare(): IShare {
$shareManager = Server::get(IManager::class);
$share = $shareManager->getShareByToken($this->token);
return $share;
}
/**
* \Sabre\DAV\Auth\Backend\AbstractBearer::challenge sets an WWW-Authenticate
* header which some DAV clients can't handle. Thus we override this function
* and make it simply return a 401.
*
* @param RequestInterface $request
* @param ResponseInterface $response
*/
#[\Override]
public function challenge(RequestInterface $request, ResponseInterface $response): void {
// Legacy ownCloud clients still authenticate via OAuth2
$enableOcClients = $this->config->getSystemValueBool('oauth2.enable_oc_clients', false);
$userAgent = $request->getHeader('User-Agent');
if ($enableOcClients && $userAgent !== null && str_contains($userAgent, 'mirall')) {
parent::challenge($request, $response);
return;
}
$response->setStatus(Http::STATUS_UNAUTHORIZED);
}
}