Skip to content

Commit

Permalink
encrypt oauth client id/secret and user tokens
Browse files Browse the repository at this point in the history
Signed-off-by: Julien Veyssier <[email protected]>
  • Loading branch information
julien-nc committed Oct 18, 2024
1 parent 4a7d3e5 commit 50ca59f
Show file tree
Hide file tree
Showing 12 changed files with 193 additions and 31 deletions.
2 changes: 1 addition & 1 deletion appinfo/info.xml
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@
<summary>Integration of GitLab software development management service</summary>
<description><![CDATA[GitLab integration provides a dashboard widget displaying your most important notifications
and a unified search provider for repositories, issues and merge requests.]]></description>
<version>3.1.2</version>
<version>3.1.3</version>
<licence>agpl</licence>
<author>Julien Veyssier</author>
<namespace>Gitlab</namespace>
Expand Down
6 changes: 3 additions & 3 deletions lib/Controller/ConfigController.php
Original file line number Diff line number Diff line change
Expand Up @@ -71,7 +71,7 @@ public function addAccount(string $url, string $token) {
$account = new GitlabAccount();
$account->setUserId($this->userId);
$account->setUrl($url);
$account->setToken($token);
$account->setEncryptedToken($token);
$account->setTokenType('personal');

try {
Expand Down Expand Up @@ -197,9 +197,9 @@ public function oauthRedirect(string $code = '', string $state = ''): RedirectRe
$account = new GitlabAccount();
$account->setUserId($this->userId);
$account->setUrl($adminOauthUrl);
$account->setToken($result['access_token']);
$account->setEncryptedToken($result['access_token']);
$account->setTokenType('oauth');
$account->setRefreshToken($result['refresh_token'] ?? '');
$account->setEncryptedRefreshToken($result['refresh_token'] ?? '');
if (isset($result['expires_in'])) {
$nowTs = (new Datetime())->getTimestamp();
$expiresAt = $nowTs + (int)$result['expires_in'];
Expand Down
6 changes: 3 additions & 3 deletions lib/Controller/GitlabAPIController.php
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,7 @@ public function __construct(
public function getUserAvatar(int $accountId, int $userId) {
try {
$account = $this->accountMapper->findById($this->userId, $accountId);
if ($account->getToken() === '') {
if ($account->getClearToken() === '') {
return new DataResponse('', 400);
}

Expand Down Expand Up @@ -83,7 +83,7 @@ public function getUserAvatar(int $accountId, int $userId) {
public function getProjectAvatar(int $accountId, int $projectId) {
try {
$account = $this->accountMapper->findById($this->userId, $accountId);
if ($account->getToken() === '') {
if ($account->getClearToken() === '') {
return new DataResponse('', 400);
}

Expand Down Expand Up @@ -117,7 +117,7 @@ public function getProjectAvatar(int $accountId, int $projectId) {
public function getTodos(int $accountId, ?string $since = null): DataResponse {
try {
$account = $this->accountMapper->findById($this->userId, $accountId);
if ($account->getToken() === '') {
if ($account->getClearToken() === '') {
return new DataResponse('', 400);
}

Expand Down
62 changes: 54 additions & 8 deletions lib/Db/GitlabAccount.php
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@

use JsonSerializable;
use OCP\AppFramework\Db\Entity;
use OCP\Security\ICrypto;

/**
* @method void setUserId(string $userId)
Expand All @@ -25,14 +26,29 @@
* @method void setUserInfoDisplayName(string $userInfoDisplayName)
*/
class GitlabAccount extends Entity implements JsonSerializable {
protected string $userId = '';
protected string $url = '';
protected string $token = '';
protected string $tokenType = '';
protected ?int $tokenExpiresAt = null;
protected ?string $refreshToken = null;
protected ?string $userInfoName = null;
protected ?string $userInfoDisplayName = null;
protected $userId;
protected $url;
protected $token;
protected $tokenType;
protected $tokenExpiresAt;
protected $refreshToken;
protected $userInfoName;
protected $userInfoDisplayName;

private ICrypto $crypto;

public function __construct() {
$this->addType('id', 'integer');
$this->addType('url', 'string');
$this->addType('token', 'string');
$this->addType('token_type', 'string');
$this->addType('token_expires_at', 'integer');
$this->addType('refresh_token', 'string');
$this->addType('user_info_name', 'string');
$this->addType('user_info_display_name', 'string');

$this->crypto = \OC::$server->get(ICrypto::class);

Check failure on line 50 in lib/Db/GitlabAccount.php

View workflow job for this annotation

GitHub Actions / Psalm check on PHP 8.0 and OCP ^26

UndefinedClass

lib/Db/GitlabAccount.php:50:19: UndefinedClass: Class, interface or enum named OC does not exist (see https://psalm.dev/019)

Check failure on line 50 in lib/Db/GitlabAccount.php

View workflow job for this annotation

GitHub Actions / Psalm check on PHP 8.1 and OCP ^26

UndefinedClass

lib/Db/GitlabAccount.php:50:19: UndefinedClass: Class, interface or enum named OC does not exist (see https://psalm.dev/019)

Check failure on line 50 in lib/Db/GitlabAccount.php

View workflow job for this annotation

GitHub Actions / Psalm check on PHP 8.2 and OCP ^26

UndefinedClass

lib/Db/GitlabAccount.php:50:19: UndefinedClass: Class, interface or enum named OC does not exist (see https://psalm.dev/019)

Check failure on line 50 in lib/Db/GitlabAccount.php

View workflow job for this annotation

GitHub Actions / Psalm check on PHP 8.3 and OCP ^26

UndefinedClass

lib/Db/GitlabAccount.php:50:19: UndefinedClass: Class, interface or enum named OC does not exist (see https://psalm.dev/019)

Check failure on line 50 in lib/Db/GitlabAccount.php

View workflow job for this annotation

GitHub Actions / Psalm check on PHP 8.0 and OCP ^27

UndefinedClass

lib/Db/GitlabAccount.php:50:19: UndefinedClass: Class, interface or enum named OC does not exist (see https://psalm.dev/019)

Check failure on line 50 in lib/Db/GitlabAccount.php

View workflow job for this annotation

GitHub Actions / Psalm check on PHP 8.1 and OCP ^27

UndefinedClass

lib/Db/GitlabAccount.php:50:19: UndefinedClass: Class, interface or enum named OC does not exist (see https://psalm.dev/019)

Check failure on line 50 in lib/Db/GitlabAccount.php

View workflow job for this annotation

GitHub Actions / Psalm check on PHP 8.3 and OCP ^27

UndefinedClass

lib/Db/GitlabAccount.php:50:19: UndefinedClass: Class, interface or enum named OC does not exist (see https://psalm.dev/019)

Check failure on line 50 in lib/Db/GitlabAccount.php

View workflow job for this annotation

GitHub Actions / Psalm check on PHP 8.0 and OCP ^28

UndefinedClass

lib/Db/GitlabAccount.php:50:19: UndefinedClass: Class, interface or enum named OC does not exist (see https://psalm.dev/019)

Check failure on line 50 in lib/Db/GitlabAccount.php

View workflow job for this annotation

GitHub Actions / Psalm check on PHP 8.1 and OCP ^28

UndefinedClass

lib/Db/GitlabAccount.php:50:19: UndefinedClass: Class, interface or enum named OC does not exist (see https://psalm.dev/019)

Check failure on line 50 in lib/Db/GitlabAccount.php

View workflow job for this annotation

GitHub Actions / Psalm check on PHP 8.2 and OCP ^28

UndefinedClass

lib/Db/GitlabAccount.php:50:19: UndefinedClass: Class, interface or enum named OC does not exist (see https://psalm.dev/019)

Check failure on line 50 in lib/Db/GitlabAccount.php

View workflow job for this annotation

GitHub Actions / Psalm check on PHP 8.3 and OCP ^28

UndefinedClass

lib/Db/GitlabAccount.php:50:19: UndefinedClass: Class, interface or enum named OC does not exist (see https://psalm.dev/019)

Check failure on line 50 in lib/Db/GitlabAccount.php

View workflow job for this annotation

GitHub Actions / Psalm check on PHP 8.0 and OCP dev-master

UndefinedClass

lib/Db/GitlabAccount.php:50:19: UndefinedClass: Class, interface or enum named OC does not exist (see https://psalm.dev/019)

Check failure on line 50 in lib/Db/GitlabAccount.php

View workflow job for this annotation

GitHub Actions / Psalm check on PHP 8.1 and OCP dev-master

UndefinedClass

lib/Db/GitlabAccount.php:50:19: UndefinedClass: Class, interface or enum named OC does not exist (see https://psalm.dev/019)

Check failure on line 50 in lib/Db/GitlabAccount.php

View workflow job for this annotation

GitHub Actions / Psalm check on PHP 8.2 and OCP dev-master

UndefinedClass

lib/Db/GitlabAccount.php:50:19: UndefinedClass: Class, interface or enum named OC does not exist (see https://psalm.dev/019)

Check failure on line 50 in lib/Db/GitlabAccount.php

View workflow job for this annotation

GitHub Actions / Psalm check on PHP 8.3 and OCP dev-master

UndefinedClass

lib/Db/GitlabAccount.php:50:19: UndefinedClass: Class, interface or enum named OC does not exist (see https://psalm.dev/019)
}

public function jsonSerialize(): array {
return [
Expand All @@ -46,4 +62,34 @@ public function jsonSerialize(): array {
'userInfoDisplayName' => $this->userInfoDisplayName,
];
}

public function getClearToken(): string {
if (is_string($this->token) && $this->token !== '') {
return $this->crypto->decrypt($this->token);
}
return $this->token;
}

public function setEncryptedToken(string $token): void {
if ($token !== '') {
$this->setter('token', [$this->crypto->encrypt($token)]);
} else {
$this->setter('token', [$token]);
}
}

public function getClearRefreshToken(): ?string {
if (is_string($this->refreshToken) && $this->refreshToken !== '') {
return $this->crypto->decrypt($this->refreshToken);
}
return $this->refreshToken;
}

public function setEncryptedRefreshToken(?string $refreshToken): void {
if (is_string($refreshToken) && $refreshToken !== '') {
$this->setter('refreshToken', [$this->crypto->encrypt($refreshToken)]);
} else {
$this->setter('refreshToken', [$refreshToken]);
}
}
}
4 changes: 2 additions & 2 deletions lib/Migration/MultiAccountRepairStep.php
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@ public function run(IOutput $output) {
$account = new GitlabAccount();
$account->setUserId($user->getUID());
$account->setUrl($this->config->getUserUrl($userId));
$account->setToken($this->config->getUserToken($userId));
$account->setEncryptedToken($this->config->getUserToken($userId));
$account->setTokenType($this->config->getUserTokenType($userId));
if (!$account->getTokenType()) {
if ($this->config->hasUserRefreshToken($userId)) {
Expand All @@ -43,7 +43,7 @@ public function run(IOutput $output) {
$account->setTokenExpiresAt($this->config->getUserTokenExpiresAt($userId));
}
if ($this->config->hasUserRefreshToken($userId)) {
$account->setRefreshToken($this->config->getUserRefreshToken($userId));
$account->setEncryptedRefreshToken($this->config->getUserRefreshToken($userId));
}
$account->setUserInfoName($this->config->getUserName($userId));
$account->setUserInfoDisplayName($this->config->getUserDisplayName($userId));
Expand Down
97 changes: 97 additions & 0 deletions lib/Migration/Version030103Date20241017150114.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,97 @@
<?php

declare(strict_types=1);

/**
* SPDX-FileCopyrightText: 2020 Nextcloud GmbH and Nextcloud contributors
* SPDX-License-Identifier: AGPL-3.0-or-later
*/
namespace OCA\Gitlab\Migration;

use Closure;
use OCA\Gitlab\AppInfo\Application;
use OCP\DB\QueryBuilder\IQueryBuilder;
use OCP\IConfig;
use OCP\IDBConnection;
use OCP\Migration\IOutput;
use OCP\Migration\SimpleMigrationStep;
use OCP\Security\ICrypto;

class Version030103Date20241017150114 extends SimpleMigrationStep {

public function __construct(
private IDBConnection $connection,
private ICrypto $crypto,
private IConfig $config,
) {
}

/**
* @param IOutput $output
* @param Closure $schemaClosure The `\Closure` returns a `ISchemaWrapper`
* @param array $options
*/
public function postSchemaChange(IOutput $output, Closure $schemaClosure, array $options) {
// app config
foreach (['client_id', 'client_secret'] as $key) {
$value = $this->config->getAppValue(Application::APP_ID, $key);
if ($value !== '') {
$encryptedValue = $this->crypto->encrypt($value);
$this->config->setAppValue(Application::APP_ID, $key, $encryptedValue);
}
}

// ----------- user tokens
$qbUpdate = $this->connection->getQueryBuilder();
$qbUpdate->update('gitlab_accounts')
->set('token', $qbUpdate->createParameter('updateToken'))
->where(
$qbUpdate->expr()->eq('id', $qbUpdate->createParameter('updateAccountId'))
);

$qbSelect = $this->connection->getQueryBuilder();
$qbSelect->select('id', 'token')
->from('gitlab_accounts')
->where(
$qbSelect->expr()->nonEmptyString('token')
);
$req = $qbSelect->executeQuery();
while ($row = $req->fetch()) {
$accountId = (int)$row['id'];
$storedClearRefreshToken = $row['token'];
$encryptedToken = $this->crypto->encrypt($storedClearRefreshToken);
$qbUpdate->setParameter('updateAccountId', $accountId, IQueryBuilder::PARAM_INT);
$qbUpdate->setParameter('updateToken', $encryptedToken, IQueryBuilder::PARAM_STR);
$qbUpdate->executeStatement();
}
$req->closeCursor();

// ----------- user refresh tokens
$qbUpdate = $this->connection->getQueryBuilder();
$qbUpdate->update('gitlab_accounts')
->set('refresh_token', $qbUpdate->createParameter('updateRefreshToken'))
->where(
$qbUpdate->expr()->eq('id', $qbUpdate->createParameter('updateAccountId'))
);

$qbSelect = $this->connection->getQueryBuilder();
$qbSelect->select('id', 'refresh_token')
->from('gitlab_accounts')
->where(
$qbSelect->expr()->nonEmptyString('refresh_token')
)
->andWhere(
$qbSelect->expr()->isNotNull('refresh_token')
);
$req = $qbSelect->executeQuery();
while ($row = $req->fetch()) {
$accountId = (int)$row['id'];
$storedClearRefreshToken = $row['refresh_token'];
$encryptedToken = $this->crypto->encrypt($storedClearRefreshToken);
$qbUpdate->setParameter('updateAccountId', $accountId, IQueryBuilder::PARAM_INT);
$qbUpdate->setParameter('updateRefreshToken', $encryptedToken, IQueryBuilder::PARAM_STR);
$qbUpdate->executeStatement();
}
$req->closeCursor();
}
}
2 changes: 1 addition & 1 deletion lib/Search/GitlabSearchIssuesProvider.php
Original file line number Diff line number Diff line change
Expand Up @@ -102,7 +102,7 @@ public function search(IUser $user, ISearchQuery $query): SearchResult {

$accounts = $this->accountMapper->find($user->getUID());
foreach ($accounts as $account) {
$accessToken = $account->getToken();
$accessToken = $account->getClearToken();
if ($accessToken === '') {
continue;
}
Expand Down
2 changes: 1 addition & 1 deletion lib/Search/GitlabSearchMergeRequestsProvider.php
Original file line number Diff line number Diff line change
Expand Up @@ -102,7 +102,7 @@ public function search(IUser $user, ISearchQuery $query): SearchResult {

$accounts = $this->accountMapper->find($user->getUID());
foreach ($accounts as $account) {
$accessToken = $account->getToken();
$accessToken = $account->getClearToken();
if ($accessToken === '') {
continue;
}
Expand Down
2 changes: 1 addition & 1 deletion lib/Search/GitlabSearchReposProvider.php
Original file line number Diff line number Diff line change
Expand Up @@ -102,7 +102,7 @@ public function search(IUser $user, ISearchQuery $query): SearchResult {

$accounts = $this->accountMapper->find($user->getUID());
foreach ($accounts as $account) {
$accessToken = $account->getToken();
$accessToken = $account->getClearToken();
if ($accessToken === '') {
continue;
}
Expand Down
27 changes: 23 additions & 4 deletions lib/Service/ConfigService.php
Original file line number Diff line number Diff line change
Expand Up @@ -7,31 +7,50 @@
use OCA\Gitlab\AppInfo\Application;
use OCP\IConfig;
use OCP\PreConditionNotMetException;
use OCP\Security\ICrypto;

class ConfigService {
public function __construct(
private IConfig $config,
private ICrypto $crypto,
) {
}

public function getEncryptedAppValue(string $key): string {
$value = $this->config->getAppValue(Application::APP_ID, $key);
if ($value === '') {
return $value;
}
return $this->crypto->decrypt($value);
}

public function setEncryptedAppValue(string $key, string $value): void {
if ($value === '') {
$this->config->setAppValue(Application::APP_ID, $key, $value);
} else {
$encryptedValue = $this->crypto->encrypt($value);
$this->config->setAppValue(Application::APP_ID, $key, $encryptedValue);
}
}

public function getAdminClientId(): string {
return $this->config->getAppValue(Application::APP_ID, 'client_id');
return $this->getEncryptedAppValue('client_id');
}

public function setAdminClientId(string $clientId): void {
$this->config->setAppValue(Application::APP_ID, 'client_id', $clientId);
$this->setEncryptedAppValue('client_id', $clientId);
}

public function hasAdminClientSecret(): bool {
return $this->getAdminClientSecret() !== '';
}

public function getAdminClientSecret(): string {
return $this->config->getAppValue(Application::APP_ID, 'client_secret');
return $this->getEncryptedAppValue('client_secret');
}

public function setAdminClientSecret(string $clientSecret): void {
$this->config->setAppValue(Application::APP_ID, 'client_secret', $clientSecret);
$this->setEncryptedAppValue('client_secret', $clientSecret);
}

public function getAdminOauthUrl(): string {
Expand Down
12 changes: 6 additions & 6 deletions lib/Service/GitlabAPIService.php
Original file line number Diff line number Diff line change
Expand Up @@ -243,7 +243,7 @@ public function request(?GitlabAccount $account, string $baseUrl, string $endPoi

// try anonymous request if no user (public page) or user not connected to a gitlab account
if ($account !== null) {
$accessToken = $account->getToken();
$accessToken = $account->getClearToken();
if ($accessToken !== '') {
$options['headers']['Authorization'] = 'Bearer ' . $accessToken;
}
Expand Down Expand Up @@ -301,7 +301,7 @@ public function request(?GitlabAccount $account, string $baseUrl, string $endPoi
}

private function checkTokenExpiration(GitlabAccount $account): void {
if ($account->getRefreshToken() && $account->getTokenExpiresAt()) {
if ($account->getClearRefreshToken() && $account->getTokenExpiresAt()) {
$nowTs = (new DateTime())->getTimestamp();
// if token expires in less than a minute or is already expired
if ($nowTs > $account->getTokenExpiresAt() - 60) {
Expand All @@ -312,7 +312,7 @@ private function checkTokenExpiration(GitlabAccount $account): void {

private function refreshToken(GitlabAccount $account): bool {
$adminOauthUrl = $this->config->getAdminOauthUrl();
$refreshToken = $account->getRefreshToken();
$refreshToken = $account->getClearRefreshToken();
if (!$refreshToken) {
$this->logger->error('No GitLab refresh token found', ['app' => Application::APP_ID]);
return false;
Expand All @@ -329,8 +329,8 @@ private function refreshToken(GitlabAccount $account): bool {
$accessToken = $result['access_token'];
$refreshToken = $result['refresh_token'];
$account->setUrl($adminOauthUrl);
$account->setToken($accessToken);
$account->setRefreshToken($refreshToken);
$account->setEncryptedToken($accessToken);
$account->setEncryptedRefreshToken($refreshToken);
if (isset($result['expires_in'])) {
$nowTs = (new DateTime())->getTimestamp();
$expiresAt = $nowTs + (int) $result['expires_in'];
Expand Down Expand Up @@ -361,7 +361,7 @@ public function revokeOauthToken(GitlabAccount $account): array {
'body' => json_encode([
'client_id' => $this->config->getAdminClientId(),
'client_secret' => $this->config->getAdminClientSecret(),
'token' => $account->getToken(),
'token' => $account->getClearToken(),
]),
];

Expand Down
2 changes: 1 addition & 1 deletion tests/unit/Controller/GitlabAPIControllerTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,7 @@ protected function setUp(): void {

$account = new GitlabAccount();
$account->setUrl('https://gitlab.com');
$account->setToken(self::API_TOKEN);
$account->setEncryptedToken(self::API_TOKEN);

$accountMapper = $this->createMock(GitlabAccountMapper::class);
$accountMapper->method('findById')->willReturn($account);
Expand Down

0 comments on commit 50ca59f

Please sign in to comment.