diff --git a/build/psalm-baseline.xml b/build/psalm-baseline.xml
index 2c2784e7fb1..b6137b7f0c1 100644
--- a/build/psalm-baseline.xml
+++ b/build/psalm-baseline.xml
@@ -2693,12 +2693,8 @@
-
-
+
-
-
-
diff --git a/lib/private/TemplateLayout.php b/lib/private/TemplateLayout.php
index 60c7526435e..2bc9ff5a2af 100644
--- a/lib/private/TemplateLayout.php
+++ b/lib/private/TemplateLayout.php
@@ -31,6 +31,8 @@ use OCP\Util;
class TemplateLayout extends \OC_Template {
private static $versionHash = '';
+ /** @var string[] */
+ private static $cacheBusterCache = [];
/** @var CSSResourceLocator|null */
public static $cssLocator = null;
@@ -38,38 +40,29 @@ class TemplateLayout extends \OC_Template {
/** @var JSResourceLocator|null */
public static $jsLocator = null;
- /** @var IConfig */
- private $config;
-
- /** @var IInitialStateService */
- private $initialState;
-
- /** @var INavigationManager */
- private $navigationManager;
+ private IConfig $config;
+ private IAppManager $appManager;
+ private InitialStateService $initialState;
+ private INavigationManager $navigationManager;
/**
* @param string $renderAs
* @param string $appId application id
*/
public function __construct($renderAs, $appId = '') {
- /** @var IConfig */
- $this->config = \OC::$server->get(IConfig::class);
-
- /** @var IInitialStateService */
- $this->initialState = \OC::$server->get(IInitialStateService::class);
+ $this->config = \OCP\Server::get(IConfig::class);
+ $this->appManager = \OCP\Server::get(IAppManager::class);
+ $this->initialState = \OCP\Server::get(InitialStateService::class);
+ $this->navigationManager = \OCP\Server::get(INavigationManager::class);
- // Add fallback theming variables if theming is disabled
- if ($renderAs !== TemplateResponse::RENDER_AS_USER
- || !\OC::$server->getAppManager()->isEnabledForUser('theming')) {
+ // Add fallback theming variables if not rendered as user
+ if ($renderAs !== TemplateResponse::RENDER_AS_USER) {
// TODO cache generated default theme if enabled for fallback if server is erroring ?
Util::addStyle('theming', 'default');
}
// Decide which page we show
if ($renderAs === TemplateResponse::RENDER_AS_USER) {
- /** @var INavigationManager */
- $this->navigationManager = \OC::$server->get(INavigationManager::class);
-
parent::__construct('core', 'layout.user');
if (in_array(\OC_App::getCurrentApp(), ['settings','admin', 'help']) !== false) {
$this->assign('bodyid', 'body-settings');
@@ -90,7 +83,7 @@ class TemplateLayout extends \OC_Template {
}
// Set body data-theme
$this->assign('enabledThemes', []);
- if (\OC::$server->getAppManager()->isEnabledForUser('theming') && class_exists('\OCA\Theming\Service\ThemesService')) {
+ if ($this->appManager->isEnabledForUser('theming') && class_exists('\OCA\Theming\Service\ThemesService')) {
/** @var \OCA\Theming\Service\ThemesService */
$themesService = \OC::$server->get(\OCA\Theming\Service\ThemesService::class);
$this->assign('enabledThemes', $themesService->getEnabledThemes());
@@ -101,8 +94,8 @@ class TemplateLayout extends \OC_Template {
$this->assign('logoUrl', $logoUrl);
// Set default entry name
- $defaultEntryId = \OCP\Server::get(INavigationManager::class)->getDefaultEntryIdForUser();
- $defaultEntry = \OCP\Server::get(INavigationManager::class)->get($defaultEntryId);
+ $defaultEntryId = $this->navigationManager->getDefaultEntryIdForUser();
+ $defaultEntry = $this->navigationManager->get($defaultEntryId);
$this->assign('defaultAppName', $defaultEntry['name']);
// Add navigation entry
@@ -182,8 +175,7 @@ class TemplateLayout extends \OC_Template {
$showSimpleSignup = true;
}
- $appManager = \OCP\Server::get(IAppManager::class);
- if ($appManager->isEnabledForUser('registration')) {
+ if ($this->appManager->isEnabledForUser('registration')) {
$urlGenerator = \OCP\Server::get(IURLGenerator::class);
$signUpLink = $urlGenerator->getAbsoluteURL('/index.php/apps/registration/');
}
@@ -203,7 +195,7 @@ class TemplateLayout extends \OC_Template {
$this->assign('locale', $locale);
$this->assign('direction', $direction);
- if (\OC::$server->getSystemConfig()->getValue('installed', false)) {
+ if ($this->config->getSystemValueBool('installed', false)) {
if (empty(self::$versionHash)) {
$v = \OC_App::getAppVersions();
$v['core'] = implode('.', \OCP\Util::getVersion());
@@ -224,7 +216,7 @@ class TemplateLayout extends \OC_Template {
\OCP\Server::get(ServerVersion::class),
\OCP\Util::getL10N('lib'),
\OCP\Server::get(Defaults::class),
- \OC::$server->getAppManager(),
+ $this->appManager,
\OC::$server->getSession(),
\OC::$server->getUserSession()->getUser(),
$this->config,
@@ -314,34 +306,52 @@ class TemplateLayout extends \OC_Template {
// allows chrome workspace mapping in debug mode
return '';
}
- $themingSuffix = '';
- $v = [];
- if ($this->config->getSystemValueBool('installed', false)) {
- if (\OC::$server->getAppManager()->isInstalled('theming')) {
- $themingSuffix = '-' . $this->config->getAppValue('theming', 'cachebuster', '0');
- }
- $v = \OC_App::getAppVersions();
+ if ($this->config->getSystemValueBool('installed', false) === false) {
+ // if not installed just return the version hash
+ return '?v=' . self::$versionHash;
}
- // Try the webroot path for a match
- if ($path !== false && $path !== '') {
- $appName = $this->getAppNamefromPath($path);
- if (array_key_exists($appName, $v)) {
- $appVersion = $v[$appName];
- return '?v=' . substr(md5($appVersion), 0, 8) . $themingSuffix;
- }
+ $hash = false;
+ // Try the web-root first
+ if (is_string($path) && $path !== '') {
+ $hash = $this->getVersionHashByPath($path);
+ }
+ // If not found try the file
+ if ($hash === false && is_string($file) && $file !== '') {
+ $hash = $this->getVersionHashByPath($file);
}
- // fallback to the file path instead
- if ($file !== false && $file !== '') {
- $appName = $this->getAppNamefromPath($file);
- if (array_key_exists($appName, $v)) {
- $appVersion = $v[$appName];
- return '?v=' . substr(md5($appVersion), 0, 8) . $themingSuffix;
+ // As a last resort we use the server version hash
+ if ($hash === false) {
+ $hash = self::$versionHash;
+ }
+
+ // The theming app is force-enabled thus the cache buster is always available
+ $themingSuffix = '-' . $this->config->getAppValue('theming', 'cachebuster', '0');
+
+ return '?v=' . $hash . $themingSuffix;
+ }
+
+ private function getVersionHashByPath(string $path): string|false {
+ if (array_key_exists($path, self::$cacheBusterCache) === false) {
+ // Not yet cached, so lets find the cache buster string
+ $appId = $this->getAppNamefromPath($path);
+ if ($appId === false || $appId === null) {
+ // No app Id could be guessed
+ return false;
+ }
+
+ $appVersion = $this->appManager->getAppVersion($appId);
+ // For shipped apps the app version is not a single source of truth, we rather also need to consider the Nextcloud version
+ if ($this->appManager->isShipped($appId)) {
+ $appVersion .= '-' . self::$versionHash;
}
+
+ $hash = substr(md5($appVersion), 0, 8);
+ self::$cacheBusterCache[$path] = $hash;
}
- return '?v=' . self::$versionHash . $themingSuffix;
+ return self::$cacheBusterCache[$path];
}
/**
diff --git a/tests/lib/TemplateLayoutTest.php b/tests/lib/TemplateLayoutTest.php
new file mode 100644
index 00000000000..405f1df7330
--- /dev/null
+++ b/tests/lib/TemplateLayoutTest.php
@@ -0,0 +1,84 @@
+createMock(IAppManager::class);
+ $appManager->expects(self::any())
+ ->method('getAppVersion')
+ ->willReturnCallback(fn ($appId) => match ($appId) {
+ 'shippedApp' => 'shipped_1',
+ 'otherApp' => 'other_2',
+ default => "$appId",
+ });
+ $appManager->expects(self::any())
+ ->method('isShipped')
+ ->willReturnCallback(fn (string $app) => $app === 'shippedApp');
+
+ $config = $this->createMock(IConfig::class);
+ $config->expects(self::atLeastOnce())
+ ->method('getSystemValueBool')
+ ->willReturnMap([
+ ['installed', false, $installed],
+ ['debug', false, $debug],
+ ]);
+ $config->expects(self::any())
+ ->method('getAppValue')
+ ->with('theming', 'cachebuster', '0')
+ ->willReturn('42');
+
+ $initialState = $this->createMock(InitialStateService::class);
+
+ $this->overwriteService(IConfig::class, $config);
+ $this->overwriteService(IAppManager::class, $appManager);
+ $this->overwriteService(InitialStateService::class, $initialState);
+
+ $layout = $this->getMockBuilder(TemplateLayout::class)
+ ->onlyMethods(['getAppNamefromPath'])
+ ->setConstructorArgs([TemplateResponse::RENDER_AS_ERROR])
+ ->getMock();
+
+ self::invokePrivate(TemplateLayout::class, 'versionHash', ['version_hash']);
+
+ $layout->expects(self::any())
+ ->method('getAppNamefromPath')
+ ->willReturnCallback(fn ($appName) => match($appName) {
+ 'apps/shipped' => 'shippedApp',
+ 'other/app.css' => 'otherApp',
+ default => false,
+ });
+
+ $hash = self::invokePrivate($layout, 'getVersionHashSuffix', [$path, $file]);
+ self::assertEquals($expected, $hash);
+ }
+
+ public static function dataVersionHash() {
+ return [
+ 'no hash if in debug mode' => ['apps/shipped', 'style.css', true, true, ''],
+ 'only version hash if not installed' => ['apps/shipped', 'style.css', false, false, '?v=version_hash'],
+ 'version hash with cache buster if app not found' => ['unknown/path', '', true, false, '?v=version_hash-42'],
+ 'version hash with cache buster if neither path nor file provided' => [false, false, true, false, '?v=version_hash-42'],
+ 'app version hash if external app' => ['', 'other/app.css', true, false, '?v=' . substr(md5('other_2'), 0, 8) . '-42'],
+ 'app version and version hash if shipped app' => ['apps/shipped', 'style.css', true, false, '?v=' . substr(md5('shipped_1-version_hash'), 0, 8) . '-42'],
+ 'prefer path over file' => ['apps/shipped', 'other/app.css', true, false, '?v=' . substr(md5('shipped_1-version_hash'), 0, 8) . '-42'],
+ ];
+ }
+
+}