Server IP : 184.154.167.98 / Your IP : 18.118.1.63 Web Server : Apache System : Linux pink.dnsnetservice.com 4.18.0-553.22.1.lve.1.el8.x86_64 #1 SMP Tue Oct 8 15:52:54 UTC 2024 x86_64 User : puertode ( 1767) PHP Version : 8.2.27 Disable Function : NONE MySQL : OFF | cURL : ON | WGET : ON | Perl : ON | Python : ON | Sudo : ON | Pkexec : ON Directory : /home/puertode/public_html/fotografico/apps/notifications/lib/Controller/ |
Upload File : |
<?php /** * @copyright Copyright (c) 2017 Joas Schilling <coding@schilljs.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\Notifications\Controller; use OC\Authentication\Exceptions\InvalidTokenException; use OC\Authentication\Token\IProvider; use OC\Authentication\Token\IToken; use OC\Security\IdentityProof\Manager; use OCP\AppFramework\Http; use OCP\AppFramework\Http\DataResponse; use OCP\AppFramework\OCSController; use OCP\DB\QueryBuilder\IQueryBuilder; use OCP\IDBConnection; use OCP\IRequest; use OCP\ISession; use OCP\IUser; use OCP\IUserSession; class PushController extends OCSController { /** @var IDBConnection */ private $db; /** @var ISession */ private $session; /** @var IUserSession */ private $userSession; /** @var IProvider */ private $tokenProvider; /** @var Manager */ private $identityProof; /** * @param string $appName * @param IRequest $request * @param IDBConnection $db * @param ISession $session * @param IUserSession $userSession * @param IProvider $tokenProvider * @param Manager $identityProof */ public function __construct($appName, IRequest $request, IDBConnection $db, ISession $session, IUserSession $userSession, IProvider $tokenProvider, Manager $identityProof) { parent::__construct($appName, $request); $this->db = $db; $this->session = $session; $this->userSession = $userSession; $this->tokenProvider = $tokenProvider; $this->identityProof = $identityProof; } /** * @NoAdminRequired * * @param string $pushTokenHash * @param string $devicePublicKey * @param string $proxyServer * @return DataResponse */ public function registerDevice(string $pushTokenHash, string $devicePublicKey, string $proxyServer): DataResponse { $user = $this->userSession->getUser(); if (!$user instanceof IUser) { return new DataResponse([], Http::STATUS_UNAUTHORIZED); } if (!preg_match('/^([a-f0-9]{128})$/', $pushTokenHash)) { return new DataResponse(['message' => 'INVALID_PUSHTOKEN_HASH'], Http::STATUS_BAD_REQUEST); } if ( strpos($devicePublicKey, '-----BEGIN PUBLIC KEY-----' . "\n") !== 0 || ((\strlen($devicePublicKey) !== 450 || strpos($devicePublicKey, "\n" . '-----END PUBLIC KEY-----') !== 425) && (\strlen($devicePublicKey) !== 451 || strpos($devicePublicKey, "\n" . '-----END PUBLIC KEY-----' . "\n") !== 425)) ) { return new DataResponse(['message' => 'INVALID_DEVICE_KEY'], Http::STATUS_BAD_REQUEST); } if ( !filter_var($proxyServer, FILTER_VALIDATE_URL) || \strlen($proxyServer) > 256 || !preg_match('/^(https\:\/\/|http\:\/\/localhost(\:\d{0,5})?\/)/', $proxyServer) ) { return new DataResponse(['message' => 'INVALID_PROXY_SERVER'], Http::STATUS_BAD_REQUEST); } $tokenId = $this->session->get('token-id'); if (!\is_int($tokenId)) { return new DataResponse(['message' => 'INVALID_SESSION_TOKEN'], Http::STATUS_BAD_REQUEST); } try { $token = $this->tokenProvider->getTokenById($tokenId); } catch (InvalidTokenException $e) { return new DataResponse(['message' => 'INVALID_SESSION_TOKEN'], Http::STATUS_BAD_REQUEST); } $key = $this->identityProof->getKey($user); $deviceIdentifier = json_encode([$user->getCloudId(), $token->getId()]); openssl_sign($deviceIdentifier, $signature, $key->getPrivate(), OPENSSL_ALGO_SHA512); $deviceIdentifier = base64_encode(hash('sha512', $deviceIdentifier, true)); $appType = 'unknown'; if ($this->request->isUserAgent([ IRequest::USER_AGENT_TALK_ANDROID, IRequest::USER_AGENT_TALK_IOS, ])) { $appType = 'talk'; } elseif ($this->request->isUserAgent([ IRequest::USER_AGENT_CLIENT_ANDROID, IRequest::USER_AGENT_CLIENT_IOS, ])) { $appType = 'nextcloud'; } $created = $this->savePushToken($user, $token, $deviceIdentifier, $devicePublicKey, $pushTokenHash, $proxyServer, $appType); return new DataResponse([ 'publicKey' => $key->getPublic(), 'deviceIdentifier' => $deviceIdentifier, 'signature' => base64_encode($signature), ], $created ? Http::STATUS_CREATED : Http::STATUS_OK); } /** * @NoAdminRequired * * @return DataResponse */ public function removeDevice(): DataResponse { $user = $this->userSession->getUser(); if (!$user instanceof IUser) { return new DataResponse([], Http::STATUS_UNAUTHORIZED); } $tokenId = (int)$this->session->get('token-id'); try { $token = $this->tokenProvider->getTokenById($tokenId); } catch (InvalidTokenException $e) { return new DataResponse(['message' => 'INVALID_SESSION_TOKEN'], Http::STATUS_BAD_REQUEST); } if ($this->deletePushToken($user, $token)) { return new DataResponse([], Http::STATUS_ACCEPTED); } return new DataResponse([], Http::STATUS_OK); } /** * @param IUser $user * @param IToken $token * @param string $deviceIdentifier * @param string $devicePublicKey * @param string $pushTokenHash * @param string $proxyServer * @param string $appType * @return bool If the hash was new to the database */ protected function savePushToken(IUser $user, IToken $token, string $deviceIdentifier, string $devicePublicKey, string $pushTokenHash, string $proxyServer, string $appType): bool { $query = $this->db->getQueryBuilder(); $query->select('*') ->from('notifications_pushtokens') ->where($query->expr()->eq('uid', $query->createNamedParameter($user->getUID()))) ->andWhere($query->expr()->eq('token', $query->createNamedParameter($token->getId()))); $result = $query->execute(); $row = $result->fetch(); $result->closeCursor(); if (!$row) { // In case the auth token is new, delete potentially old entries for the same device (push token) by this user $this->deletePushTokenByHash($user, $pushTokenHash); return $this->insertPushToken($user, $token, $deviceIdentifier, $devicePublicKey, $pushTokenHash, $proxyServer, $appType); } return $this->updatePushToken($user, $token, $devicePublicKey, $pushTokenHash, $proxyServer, $appType); } /** * @param IUser $user * @param IToken $token * @param string $deviceIdentifier * @param string $devicePublicKey * @param string $pushTokenHash * @param string $proxyServer * @param string $appType * @return bool If the entry was created */ protected function insertPushToken(IUser $user, IToken $token, string $deviceIdentifier, string $devicePublicKey, string $pushTokenHash, string $proxyServer, string $appType): bool { $devicePublicKeyHash = hash('sha512', $devicePublicKey); $query = $this->db->getQueryBuilder(); $query->insert('notifications_pushtokens') ->values([ 'uid' => $query->createNamedParameter($user->getUID()), 'token' => $query->createNamedParameter($token->getId(), IQueryBuilder::PARAM_INT), 'deviceidentifier' => $query->createNamedParameter($deviceIdentifier), 'devicepublickey' => $query->createNamedParameter($devicePublicKey), 'devicepublickeyhash' => $query->createNamedParameter($devicePublicKeyHash), 'pushtokenhash' => $query->createNamedParameter($pushTokenHash), 'proxyserver' => $query->createNamedParameter($proxyServer), 'apptype' => $query->createNamedParameter($appType), ]); return $query->execute() > 0; } /** * @param IUser $user * @param IToken $token * @param string $devicePublicKey * @param string $pushTokenHash * @param string $proxyServer * @param string $appType * @return bool If the entry was updated */ protected function updatePushToken(IUser $user, IToken $token, string $devicePublicKey, string $pushTokenHash, string $proxyServer, string $appType): bool { $devicePublicKeyHash = hash('sha512', $devicePublicKey); $query = $this->db->getQueryBuilder(); $query->update('notifications_pushtokens') ->set('devicepublickey', $query->createNamedParameter($devicePublicKey)) ->set('devicepublickeyhash', $query->createNamedParameter($devicePublicKeyHash)) ->set('pushtokenhash', $query->createNamedParameter($pushTokenHash)) ->set('proxyserver', $query->createNamedParameter($proxyServer)) ->set('apptype', $query->createNamedParameter($appType)) ->where($query->expr()->eq('uid', $query->createNamedParameter($user->getUID()))) ->andWhere($query->expr()->eq('token', $query->createNamedParameter($token->getId(), IQueryBuilder::PARAM_INT))); return $query->execute() !== 0; } /** * @param IUser $user * @param IToken $token * @return bool If the entry was deleted */ protected function deletePushToken(IUser $user, IToken $token): bool { $query = $this->db->getQueryBuilder(); $query->delete('notifications_pushtokens') ->where($query->expr()->eq('uid', $query->createNamedParameter($user->getUID()))) ->andWhere($query->expr()->eq('token', $query->createNamedParameter($token->getId(), IQueryBuilder::PARAM_INT))); return $query->execute() !== 0; } /** * @param IUser $user * @param string $pushTokenHash * @return bool If the entry was deleted */ protected function deletePushTokenByHash(IUser $user, string $pushTokenHash): bool { $query = $this->db->getQueryBuilder(); $query->delete('notifications_pushtokens') ->where($query->expr()->eq('uid', $query->createNamedParameter($user->getUID()))) ->andWhere($query->expr()->eq('pushtokenhash', $query->createNamedParameter($pushTokenHash))); return $query->execute() !== 0; } }