From 2fccd090b398d5edfd6772013e48dae90cccae24 Mon Sep 17 00:00:00 2001 From: Joas Schilling Date: Mon, 11 Dec 2023 08:59:45 +0100 Subject: fix(manifest): Check if app exists instead of accessing null as an array Signed-off-by: Joas Schilling --- apps/theming/lib/Controller/IconController.php | 30 ++++++++++++++-------- apps/theming/lib/Controller/ThemingController.php | 12 +++++++-- apps/theming/openapi.json | 10 ++++++++ .../tests/Controller/IconControllerTest.php | 7 ++++- core/templates/layout.guest.php | 2 +- core/templates/layout.public.php | 2 +- core/templates/layout.user.php | 2 +- 7 files changed, 48 insertions(+), 17 deletions(-) diff --git a/apps/theming/lib/Controller/IconController.php b/apps/theming/lib/Controller/IconController.php index 6ad67c4667a..216ca88d375 100644 --- a/apps/theming/lib/Controller/IconController.php +++ b/apps/theming/lib/Controller/IconController.php @@ -32,6 +32,7 @@ use OC\IntegrityCheck\Helpers\FileAccessHelper; use OCA\Theming\IconBuilder; use OCA\Theming\ImageManager; use OCA\Theming\ThemingDefaults; +use OCP\App\IAppManager; use OCP\AppFramework\Controller; use OCP\AppFramework\Http; use OCP\AppFramework\Http\DataDisplayResponse; @@ -50,24 +51,17 @@ class IconController extends Controller { private $imageManager; /** @var FileAccessHelper */ private $fileAccessHelper; + /** @var IAppManager */ + private $appManager; - /** - * IconController constructor. - * - * @param string $appName - * @param IRequest $request - * @param ThemingDefaults $themingDefaults - * @param IconBuilder $iconBuilder - * @param ImageManager $imageManager - * @param FileAccessHelper $fileAccessHelper - */ public function __construct( $appName, IRequest $request, ThemingDefaults $themingDefaults, IconBuilder $iconBuilder, ImageManager $imageManager, - FileAccessHelper $fileAccessHelper + FileAccessHelper $fileAccessHelper, + IAppManager $appManager ) { parent::__construct($appName, $request); @@ -75,6 +69,7 @@ class IconController extends Controller { $this->iconBuilder = $iconBuilder; $this->imageManager = $imageManager; $this->fileAccessHelper = $fileAccessHelper; + $this->appManager = $appManager; } /** @@ -92,6 +87,11 @@ class IconController extends Controller { * 404: Themed icon not found */ public function getThemedIcon(string $app, string $image): Response { + if ($app !== 'core' && !$this->appManager->isEnabledForUser($app)) { + $app = 'core'; + $image = 'favicon.png'; + } + $color = $this->themingDefaults->getColorPrimary(); try { $iconFileName = $this->imageManager->getCachedImage('icon-' . $app . '-' . $color . str_replace('/', '_', $image)); @@ -121,6 +121,10 @@ class IconController extends Controller { * 404: Favicon not found */ public function getFavicon(string $app = 'core'): Response { + if ($app !== 'core' && !$this->appManager->isEnabledForUser($app)) { + $app = 'core'; + } + $response = null; $iconFile = null; try { @@ -163,6 +167,10 @@ class IconController extends Controller { * 404: Touch icon not found */ public function getTouchIcon(string $app = 'core'): Response { + if ($app !== 'core' && !$this->appManager->isEnabledForUser($app)) { + $app = 'core'; + } + $response = null; try { $iconFile = $this->imageManager->getImage('favicon'); diff --git a/apps/theming/lib/Controller/ThemingController.php b/apps/theming/lib/Controller/ThemingController.php index b4bf6d1c3cd..91012d1e37a 100644 --- a/apps/theming/lib/Controller/ThemingController.php +++ b/apps/theming/lib/Controller/ThemingController.php @@ -445,16 +445,18 @@ class ThemingController extends Controller { /** * @NoCSRFRequired * @PublicPage + * @BruteForceProtection(action=manifest) * * Get the manifest for an app * * @param string $app ID of the app * @psalm-suppress LessSpecificReturnStatement The content of the Manifest doesn't need to be described in the return type - * @return JSONResponse + * @return JSONResponse|JSONResponse * * 200: Manifest returned + * 404: App not found */ - public function getManifest(string $app) { + public function getManifest(string $app): JSONResponse { $cacheBusterValue = $this->config->getAppValue('theming', 'cachebuster', '0'); if ($app === 'core' || $app === 'settings') { $name = $this->themingDefaults->getName(); @@ -462,6 +464,12 @@ class ThemingController extends Controller { $startUrl = $this->urlGenerator->getBaseUrl(); $description = $this->themingDefaults->getSlogan(); } else { + if (!$this->appManager->isEnabledForUser($app)) { + $response = new JSONResponse([], Http::STATUS_NOT_FOUND); + $response->throttle(['action' => 'manifest', 'app' => $app]); + return $response; + } + $info = $this->appManager->getAppInfo($app, false, $this->l10n->getLanguageCode()); $name = $info['name'] . ' - ' . $this->themingDefaults->getName(); $shortName = $info['name']; diff --git a/apps/theming/openapi.json b/apps/theming/openapi.json index f85f916dcaa..439e14fcc60 100644 --- a/apps/theming/openapi.json +++ b/apps/theming/openapi.json @@ -386,6 +386,16 @@ } } } + }, + "404": { + "description": "App not found", + "content": { + "application/json": { + "schema": { + "type": "object" + } + } + } } } } diff --git a/apps/theming/tests/Controller/IconControllerTest.php b/apps/theming/tests/Controller/IconControllerTest.php index 470709a3fab..d2b52cf738a 100644 --- a/apps/theming/tests/Controller/IconControllerTest.php +++ b/apps/theming/tests/Controller/IconControllerTest.php @@ -33,6 +33,7 @@ use OCA\Theming\Controller\IconController; use OCA\Theming\IconBuilder; use OCA\Theming\ImageManager; use OCA\Theming\ThemingDefaults; +use OCP\App\IAppManager; use OCP\AppFramework\Http; use OCP\AppFramework\Http\DataDisplayResponse; use OCP\AppFramework\Http\FileDisplayResponse; @@ -57,6 +58,8 @@ class IconControllerTest extends TestCase { private $iconBuilder; /** @var FileAccessHelper|\PHPUnit\Framework\MockObject\MockObject */ private $fileAccessHelper; + /** @var IAppManager|\PHPUnit\Framework\MockObject\MockObject */ + private $appManager; /** @var ImageManager */ private $imageManager; @@ -66,6 +69,7 @@ class IconControllerTest extends TestCase { $this->iconBuilder = $this->createMock(IconBuilder::class); $this->imageManager = $this->createMock(ImageManager::class); $this->fileAccessHelper = $this->createMock(FileAccessHelper::class); + $this->appManager = $this->createMock(IAppManager::class); $this->timeFactory = $this->createMock(ITimeFactory::class); $this->timeFactory->expects($this->any()) @@ -80,7 +84,8 @@ class IconControllerTest extends TestCase { $this->themingDefaults, $this->iconBuilder, $this->imageManager, - $this->fileAccessHelper + $this->fileAccessHelper, + $this->appManager, ); parent::setUp(); diff --git a/core/templates/layout.guest.php b/core/templates/layout.guest.php index eed6af129d8..5ebada92934 100644 --- a/core/templates/layout.guest.php +++ b/core/templates/layout.guest.php @@ -20,7 +20,7 @@ p($theme->getTitle()); - + diff --git a/core/templates/layout.public.php b/core/templates/layout.public.php index cd26fcba6de..81d11adffb5 100644 --- a/core/templates/layout.public.php +++ b/core/templates/layout.public.php @@ -21,7 +21,7 @@ p($theme->getTitle()); - + diff --git a/core/templates/layout.user.php b/core/templates/layout.user.php index 3b11205037e..da837ce5ff2 100644 --- a/core/templates/layout.user.php +++ b/core/templates/layout.user.php @@ -37,7 +37,7 @@ p($theme->getTitle()); - + -- cgit v1.2.3