mirror of
https://github.com/nextcloud/server.git
synced 2026-05-28 04:32:30 -04:00
Merge pull request #50233 from nextcloud/fix/update-notification
fix(updatenotification): Parse pre-release version of apps
This commit is contained in:
commit
0a3cf3caf3
3 changed files with 203 additions and 2 deletions
|
|
@ -153,14 +153,23 @@ class APIController extends OCSController {
|
|||
* @param string $appId App to search changelog entry for
|
||||
* @param string|null $version The version to search the changelog entry for (defaults to the latest installed)
|
||||
*
|
||||
* @return DataResponse<Http::STATUS_OK, array{appName: string, content: string, version: string}, array{}>|DataResponse<Http::STATUS_NOT_FOUND, array{}, array{}>
|
||||
* @return DataResponse<Http::STATUS_OK, array{appName: string, content: string, version: string}, array{}>|DataResponse<Http::STATUS_NOT_FOUND, array{}, array{}>|DataResponse<Http::STATUS_BAD_REQUEST, array{}, array{}>
|
||||
*
|
||||
* 200: Changelog entry returned
|
||||
* 400: The `version` parameter is not a valid version format
|
||||
* 404: No changelog found
|
||||
*/
|
||||
public function getAppChangelogEntry(string $appId, ?string $version = null): DataResponse {
|
||||
$version = $version ?? $this->appManager->getAppVersion($appId);
|
||||
$changes = $this->manager->getChangelog($appId, $version);
|
||||
// handle pre-release versions
|
||||
$matches = [];
|
||||
$result = preg_match('/^(\d+\.\d+(\.\d+)?)/', $version, $matches);
|
||||
if ($result === false || $result === 0) {
|
||||
return new DataResponse([], Http::STATUS_BAD_REQUEST);
|
||||
}
|
||||
$shortVersion = $matches[0];
|
||||
|
||||
$changes = $this->manager->getChangelog($appId, $shortVersion);
|
||||
|
||||
if ($changes === null) {
|
||||
return new DataResponse([], Http::STATUS_NOT_FOUND);
|
||||
|
|
|
|||
|
|
@ -338,6 +338,36 @@
|
|||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"400": {
|
||||
"description": "The `version` parameter is not a valid version format",
|
||||
"content": {
|
||||
"application/json": {
|
||||
"schema": {
|
||||
"type": "object",
|
||||
"required": [
|
||||
"ocs"
|
||||
],
|
||||
"properties": {
|
||||
"ocs": {
|
||||
"type": "object",
|
||||
"required": [
|
||||
"meta",
|
||||
"data"
|
||||
],
|
||||
"properties": {
|
||||
"meta": {
|
||||
"$ref": "#/components/schemas/OCSMeta"
|
||||
},
|
||||
"data": {
|
||||
"type": "object"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
|||
162
apps/updatenotification/tests/Controller/APIControllerTest.php
Normal file
162
apps/updatenotification/tests/Controller/APIControllerTest.php
Normal file
|
|
@ -0,0 +1,162 @@
|
|||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
/**
|
||||
* SPDX-FileCopyrightText: 2025 Nextcloud GmbH and Nextcloud contributors
|
||||
* SPDX-License-Identifier: AGPL-3.0-or-later
|
||||
*/
|
||||
namespace OCA\UpdateNotification\Tests\Controller;
|
||||
|
||||
use OC\App\AppStore\Fetcher\AppFetcher;
|
||||
use OCA\UpdateNotification\AppInfo\Application;
|
||||
use OCA\UpdateNotification\Controller\APIController;
|
||||
use OCA\UpdateNotification\Manager;
|
||||
use OCP\App\IAppManager;
|
||||
use OCP\AppFramework\Http;
|
||||
use OCP\IConfig;
|
||||
use OCP\IRequest;
|
||||
use OCP\IUserSession;
|
||||
use OCP\L10N\IFactory;
|
||||
use PHPUnit\Framework\MockObject\MockObject;
|
||||
use Test\TestCase;
|
||||
|
||||
class APIControllerTest extends TestCase {
|
||||
private IRequest&MockObject $request;
|
||||
private IConfig&MockObject $config;
|
||||
private IAppManager&MockObject $appManager;
|
||||
private AppFetcher&MockObject $appFetcher;
|
||||
private IFactory&MockObject $l10nFactory;
|
||||
private IUserSession&MockObject $userSession;
|
||||
private Manager&MockObject $manager;
|
||||
|
||||
private APIController $apiController;
|
||||
|
||||
protected function setUp(): void {
|
||||
parent::setUp();
|
||||
|
||||
$this->request = $this->createMock(IRequest::class);
|
||||
$this->config = $this->createMock(IConfig::class);
|
||||
$this->appManager = $this->createMock(IAppManager::class);
|
||||
$this->appFetcher = $this->createMock(AppFetcher::class);
|
||||
$this->l10nFactory = $this->createMock(IFactory::class);
|
||||
$this->userSession = $this->createMock(IUserSession::class);
|
||||
$this->manager = $this->createMock(Manager::class);
|
||||
|
||||
$this->apiController = new APIController(
|
||||
Application::APP_NAME,
|
||||
$this->request,
|
||||
$this->config,
|
||||
$this->appManager,
|
||||
$this->appFetcher,
|
||||
$this->l10nFactory,
|
||||
$this->userSession,
|
||||
$this->manager,
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* @dataProvider dataGetAppChangelog
|
||||
*/
|
||||
public function testGetAppChangelogEntry(
|
||||
array $params,
|
||||
bool $hasChanges,
|
||||
array $appInfo,
|
||||
array $expected,
|
||||
): void {
|
||||
$this->appManager->method('getAppInfo')
|
||||
->with('the-app')
|
||||
->willReturn($appInfo);
|
||||
$this->appManager->method('getAppVersion')
|
||||
->with('the-app')
|
||||
->willReturn($appInfo['version']);
|
||||
$this->manager->method('getChangelog')
|
||||
->with('the-app', self::anything())
|
||||
->willReturnCallback(fn ($app, $version) => $hasChanges ? "$app v$version" : null);
|
||||
|
||||
$result = $this->apiController->getAppChangelogEntry(...$params);
|
||||
$this->assertEquals($result->getStatus(), $expected['status']);
|
||||
$this->assertEquals($result->getData(), $expected['data']);
|
||||
}
|
||||
|
||||
public static function dataGetAppChangelog(): array {
|
||||
return [
|
||||
'no changes found' => [
|
||||
['the-app', null],
|
||||
false,
|
||||
[
|
||||
'name' => 'Localized name',
|
||||
'version' => '1.0.0',
|
||||
],
|
||||
[
|
||||
'status' => Http::STATUS_NOT_FOUND,
|
||||
'data' => [],
|
||||
]
|
||||
],
|
||||
'changes with version parameter' => [
|
||||
['the-app', '1.0.0'],
|
||||
true,
|
||||
[
|
||||
'name' => 'Localized name',
|
||||
'version' => '1.2.0', // installed version
|
||||
],
|
||||
[
|
||||
'status' => Http::STATUS_OK,
|
||||
'data' => [
|
||||
'appName' => 'Localized name',
|
||||
'content' => 'the-app v1.0.0',
|
||||
'version' => '1.0.0',
|
||||
],
|
||||
]
|
||||
],
|
||||
'changes without version parameter' => [
|
||||
['the-app', null],
|
||||
true,
|
||||
[
|
||||
'name' => 'Localized name',
|
||||
'version' => '1.2.0',
|
||||
],
|
||||
[
|
||||
'status' => Http::STATUS_OK,
|
||||
'data' => [
|
||||
'appName' => 'Localized name',
|
||||
'content' => 'the-app v1.2.0',
|
||||
'version' => '1.2.0',
|
||||
],
|
||||
]
|
||||
],
|
||||
'changes of pre-release version' => [
|
||||
['the-app', null],
|
||||
true,
|
||||
[
|
||||
'name' => 'Localized name',
|
||||
'version' => '1.2.0-alpha.1',
|
||||
],
|
||||
[
|
||||
'status' => Http::STATUS_OK,
|
||||
'data' => [
|
||||
'appName' => 'Localized name',
|
||||
'content' => 'the-app v1.2.0',
|
||||
'version' => '1.2.0-alpha.1',
|
||||
],
|
||||
]
|
||||
],
|
||||
'changes of pre-release version as parameter' => [
|
||||
['the-app', '1.2.0-alpha.2'],
|
||||
true,
|
||||
[
|
||||
'name' => 'Localized name',
|
||||
'version' => '1.2.0-beta.3',
|
||||
],
|
||||
[
|
||||
'status' => Http::STATUS_OK,
|
||||
'data' => [
|
||||
'appName' => 'Localized name',
|
||||
'content' => 'the-app v1.2.0',
|
||||
'version' => '1.2.0-alpha.2',
|
||||
],
|
||||
]
|
||||
],
|
||||
];
|
||||
}
|
||||
}
|
||||
Loading…
Reference in a new issue