Browse Source

Merge pull request #47927 from nextcloud/fix/migrate-away-from-oc_app

Migrate away from OC_App to IAppManager
pull/47954/head
Côme Chilliet 1 year ago
committed by GitHub
parent
commit
bcb4e781a4
No known key found for this signature in database GPG Key ID: B5690EEEBB952194
  1. 7
      apps/provisioning_api/lib/Controller/AppConfigController.php
  2. 27
      apps/provisioning_api/tests/Controller/AppConfigControllerTest.php
  3. 11
      apps/settings/lib/Controller/AppSettingsController.php
  4. 31
      apps/theming/tests/UtilTest.php
  5. 2
      core/Command/App/Enable.php
  6. 2
      core/Command/App/GetPath.php
  7. 2
      core/Command/App/Update.php
  8. 13
      core/Command/Db/Migrations/GenerateCommand.php
  9. 2
      core/Command/Db/Migrations/GenerateMetadataCommand.php
  10. 9
      core/Command/L10n/CreateJs.php
  11. 40
      lib/private/App/AppManager.php
  12. 30
      lib/private/Authentication/TwoFactorAuth/ProviderLoader.php
  13. 4
      lib/private/IntegrityCheck/Checker.php
  14. 9
      lib/private/IntegrityCheck/Helpers/AppLocator.php
  15. 2
      lib/private/L10N/Factory.php
  16. 26
      lib/private/Route/Router.php
  17. 2
      lib/private/Server.php
  18. 31
      lib/private/legacy/OC_App.php
  19. 24
      lib/public/App/IAppManager.php
  20. 19
      remote.php
  21. 11
      tests/apps.php
  22. 4
      tests/lib/App/AppManagerTest.php
  23. 12
      tests/lib/InfoXmlTest.php
  24. 4
      tests/lib/IntegrityCheck/CheckerTest.php
  25. 4
      tests/lib/IntegrityCheck/Helpers/AppLocatorTest.php
  26. 14
      tests/lib/Route/RouterTest.php

7
apps/provisioning_api/lib/Controller/AppConfigController.php

@ -10,6 +10,7 @@ namespace OCA\Provisioning_API\Controller;
use OC\AppConfig; use OC\AppConfig;
use OC\AppFramework\Middleware\Security\Exceptions\NotAdminException; use OC\AppFramework\Middleware\Security\Exceptions\NotAdminException;
use OCP\App\IAppManager;
use OCP\AppFramework\Http; use OCP\AppFramework\Http;
use OCP\AppFramework\Http\Attribute\NoAdminRequired; use OCP\AppFramework\Http\Attribute\NoAdminRequired;
use OCP\AppFramework\Http\Attribute\PasswordConfirmationRequired; use OCP\AppFramework\Http\Attribute\PasswordConfirmationRequired;
@ -35,6 +36,7 @@ class AppConfigController extends OCSController {
private IL10N $l10n, private IL10N $l10n,
private IGroupManager $groupManager, private IGroupManager $groupManager,
private IManager $settingManager, private IManager $settingManager,
private IAppManager $appManager,
) { ) {
parent::__construct($appName, $request); parent::__construct($appName, $request);
} }
@ -171,11 +173,10 @@ class AppConfigController extends OCSController {
} }
/** /**
* @param string $app
* @throws \InvalidArgumentException * @throws \InvalidArgumentException
*/ */
protected function verifyAppId(string $app) {
if (\OC_App::cleanAppId($app) !== $app) {
protected function verifyAppId(string $app): void {
if ($this->appManager->cleanAppId($app) !== $app) {
throw new \InvalidArgumentException('Invalid app id given'); throw new \InvalidArgumentException('Invalid app id given');
} }
} }

27
apps/provisioning_api/tests/Controller/AppConfigControllerTest.php

@ -7,6 +7,7 @@ namespace OCA\Provisioning_API\Tests\Controller;
use OC\AppConfig; use OC\AppConfig;
use OCA\Provisioning_API\Controller\AppConfigController; use OCA\Provisioning_API\Controller\AppConfigController;
use OCP\App\IAppManager;
use OCP\AppFramework\Http; use OCP\AppFramework\Http;
use OCP\AppFramework\Http\DataResponse; use OCP\AppFramework\Http\DataResponse;
use OCP\Exceptions\AppConfigUnknownKeyException; use OCP\Exceptions\AppConfigUnknownKeyException;
@ -17,6 +18,7 @@ use OCP\IRequest;
use OCP\IUser; use OCP\IUser;
use OCP\IUserSession; use OCP\IUserSession;
use OCP\Settings\IManager; use OCP\Settings\IManager;
use PHPUnit\Framework\MockObject\MockObject;
use Test\TestCase; use Test\TestCase;
use function json_decode; use function json_decode;
use function json_encode; use function json_encode;
@ -28,16 +30,12 @@ use function json_encode;
*/ */
class AppConfigControllerTest extends TestCase { class AppConfigControllerTest extends TestCase {
/** @var IAppConfig|\PHPUnit\Framework\MockObject\MockObject */
private $appConfig;
/** @var IUserSession|\PHPUnit\Framework\MockObject\MockObject */
private $userSession;
/** @var IL10N|\PHPUnit\Framework\MockObject\MockObject */
private $l10n;
/** @var IManager|\PHPUnit\Framework\MockObject\MockObject */
private $settingManager;
/** @var IGroupManager|\PHPUnit\Framework\MockObject\MockObject */
private $groupManager;
private IAppConfig&MockObject $appConfig;
private IUserSession&MockObject $userSession;
private IL10N&MockObject $l10n;
private IManager&MockObject $settingManager;
private IGroupManager&MockObject $groupManager;
private IAppManager $appManager;
protected function setUp(): void { protected function setUp(): void {
parent::setUp(); parent::setUp();
@ -45,8 +43,9 @@ class AppConfigControllerTest extends TestCase {
$this->appConfig = $this->createMock(AppConfig::class); $this->appConfig = $this->createMock(AppConfig::class);
$this->userSession = $this->createMock(IUserSession::class); $this->userSession = $this->createMock(IUserSession::class);
$this->l10n = $this->createMock(IL10N::class); $this->l10n = $this->createMock(IL10N::class);
$this->groupManager = $this->createMock(IGroupManager::class);
$this->settingManager = $this->createMock(IManager::class); $this->settingManager = $this->createMock(IManager::class);
$this->groupManager = $this->createMock(IGroupManager::class);
$this->appManager = \OCP\Server::get(IAppManager::class);
} }
/** /**
@ -64,7 +63,8 @@ class AppConfigControllerTest extends TestCase {
$this->userSession, $this->userSession,
$this->l10n, $this->l10n,
$this->groupManager, $this->groupManager,
$this->settingManager
$this->settingManager,
$this->appManager,
); );
} else { } else {
return $this->getMockBuilder(AppConfigController::class) return $this->getMockBuilder(AppConfigController::class)
@ -75,7 +75,8 @@ class AppConfigControllerTest extends TestCase {
$this->userSession, $this->userSession,
$this->l10n, $this->l10n,
$this->groupManager, $this->groupManager,
$this->settingManager
$this->settingManager,
$this->appManager,
]) ])
->setMethods($methods) ->setMethods($methods)
->getMock(); ->getMock();

11
apps/settings/lib/Controller/AppSettingsController.php

@ -14,7 +14,6 @@ use OC\App\AppStore\Version\VersionParser;
use OC\App\DependencyAnalyzer; use OC\App\DependencyAnalyzer;
use OC\App\Platform; use OC\App\Platform;
use OC\Installer; use OC\Installer;
use OC_App;
use OCP\App\AppPathNotFoundException; use OCP\App\AppPathNotFoundException;
use OCP\App\IAppManager; use OCP\App\IAppManager;
use OCP\AppFramework\Controller; use OCP\AppFramework\Controller;
@ -479,7 +478,7 @@ class AppSettingsController extends Controller {
$updateRequired = false; $updateRequired = false;
foreach ($appIds as $appId) { foreach ($appIds as $appId) {
$appId = OC_App::cleanAppId($appId);
$appId = $this->appManager->cleanAppId($appId);
// Check if app is already downloaded // Check if app is already downloaded
/** @var Installer $installer */ /** @var Installer $installer */
@ -537,7 +536,7 @@ class AppSettingsController extends Controller {
public function disableApps(array $appIds): JSONResponse { public function disableApps(array $appIds): JSONResponse {
try { try {
foreach ($appIds as $appId) { foreach ($appIds as $appId) {
$appId = OC_App::cleanAppId($appId);
$appId = $this->appManager->cleanAppId($appId);
$this->appManager->disableApp($appId); $this->appManager->disableApp($appId);
} }
return new JSONResponse([]); return new JSONResponse([]);
@ -553,7 +552,7 @@ class AppSettingsController extends Controller {
*/ */
#[PasswordConfirmationRequired] #[PasswordConfirmationRequired]
public function uninstallApp(string $appId): JSONResponse { public function uninstallApp(string $appId): JSONResponse {
$appId = OC_App::cleanAppId($appId);
$appId = $this->appManager->cleanAppId($appId);
$result = $this->installer->removeApp($appId); $result = $this->installer->removeApp($appId);
if ($result !== false) { if ($result !== false) {
$this->appManager->clearAppsCache(); $this->appManager->clearAppsCache();
@ -567,7 +566,7 @@ class AppSettingsController extends Controller {
* @return JSONResponse * @return JSONResponse
*/ */
public function updateApp(string $appId): JSONResponse { public function updateApp(string $appId): JSONResponse {
$appId = OC_App::cleanAppId($appId);
$appId = $this->appManager->cleanAppId($appId);
$this->config->setSystemValue('maintenance', true); $this->config->setSystemValue('maintenance', true);
try { try {
@ -594,7 +593,7 @@ class AppSettingsController extends Controller {
} }
public function force(string $appId): JSONResponse { public function force(string $appId): JSONResponse {
$appId = OC_App::cleanAppId($appId);
$appId = $this->appManager->cleanAppId($appId);
$this->appManager->ignoreNextcloudRequirementForApp($appId); $this->appManager->ignoreNextcloudRequirementForApp($appId);
return new JSONResponse(); return new JSONResponse();
} }

31
apps/theming/tests/UtilTest.php

@ -18,22 +18,17 @@ use Test\TestCase;
class UtilTest extends TestCase { class UtilTest extends TestCase {
/** @var Util */
protected $util;
/** @var IConfig|MockObject */
protected $config;
/** @var IAppData|MockObject */
protected $appData;
/** @var IAppManager|MockObject */
protected $appManager;
/** @var ImageManager|MockObject */
protected $imageManager;
protected Util $util;
protected IConfig&MockObject $config;
protected IAppData&MockObject $appData;
protected IAppManager $appManager;
protected ImageManager&MockObject $imageManager;
protected function setUp(): void { protected function setUp(): void {
parent::setUp(); parent::setUp();
$this->config = $this->createMock(IConfig::class); $this->config = $this->createMock(IConfig::class);
$this->appData = $this->createMock(IAppData::class); $this->appData = $this->createMock(IAppData::class);
$this->appManager = $this->createMock(IAppManager::class);
$this->appManager = \OCP\Server::get(IAppManager::class);
$this->imageManager = $this->createMock(ImageManager::class); $this->imageManager = $this->createMock(ImageManager::class);
$this->util = new Util($this->config, $this->appManager, $this->appData, $this->imageManager); $this->util = new Util($this->config, $this->appManager, $this->appData, $this->imageManager);
} }
@ -152,19 +147,15 @@ class UtilTest extends TestCase {
->method('getFolder') ->method('getFolder')
->with('global/images') ->with('global/images')
->willThrowException(new NotFoundException()); ->willThrowException(new NotFoundException());
$this->appManager->expects($this->once())
->method('getAppPath')
->with($app)
->willReturn(\OC_App::getAppPath($app));
$icon = $this->util->getAppIcon($app); $icon = $this->util->getAppIcon($app);
$this->assertEquals($expected, $icon); $this->assertEquals($expected, $icon);
} }
public function dataGetAppIcon() { public function dataGetAppIcon() {
return [ return [
['user_ldap', \OC_App::getAppPath('user_ldap') . '/img/app.svg'],
['user_ldap', \OCP\Server::get(IAppManager::class)->getAppPath('user_ldap') . '/img/app.svg'],
['noapplikethis', \OC::$SERVERROOT . '/core/img/logo/logo.svg'], ['noapplikethis', \OC::$SERVERROOT . '/core/img/logo/logo.svg'],
['comments', \OC_App::getAppPath('comments') . '/img/comments.svg'],
['comments', \OCP\Server::get(IAppManager::class)->getAppPath('comments') . '/img/comments.svg'],
]; ];
} }
@ -187,12 +178,6 @@ class UtilTest extends TestCase {
* @dataProvider dataGetAppImage * @dataProvider dataGetAppImage
*/ */
public function testGetAppImage($app, $image, $expected) { public function testGetAppImage($app, $image, $expected) {
if ($app !== 'core') {
$this->appManager->expects($this->once())
->method('getAppPath')
->with($app)
->willReturn(\OC_App::getAppPath($app));
}
$this->assertEquals($expected, $this->util->getAppImage($app, $image)); $this->assertEquals($expected, $this->util->getAppImage($app, $image));
} }

2
core/Command/App/Enable.php

@ -145,7 +145,7 @@ class Enable extends Command implements CompletionAwareInterface {
*/ */
public function completeArgumentValues($argumentName, CompletionContext $context): array { public function completeArgumentValues($argumentName, CompletionContext $context): array {
if ($argumentName === 'app-id') { if ($argumentName === 'app-id') {
$allApps = \OC_App::getAllApps();
$allApps = $this->appManager->getAllAppsInAppsFolders();
return array_diff($allApps, \OC_App::getEnabledApps(true, true)); return array_diff($allApps, \OC_App::getEnabledApps(true, true));
} }
return []; return [];

2
core/Command/App/GetPath.php

@ -63,7 +63,7 @@ class GetPath extends Base {
*/ */
public function completeArgumentValues($argumentName, CompletionContext $context): array { public function completeArgumentValues($argumentName, CompletionContext $context): array {
if ($argumentName === 'app') { if ($argumentName === 'app') {
return \OC_App::getAllApps();
return $this->appManager->getAllAppsInAppsFolders();
} }
return []; return [];
} }

2
core/Command/App/Update.php

@ -69,7 +69,7 @@ class Update extends Command {
return 1; return 1;
} }
} elseif ($input->getOption('all') || $input->getOption('showonly')) { } elseif ($input->getOption('all') || $input->getOption('showonly')) {
$apps = \OC_App::getAllApps();
$apps = $this->manager->getAllAppsInAppsFolders();
} else { } else {
$output->writeln('<error>Please specify an app to update or "--all" to update all updatable apps"</error>'); $output->writeln('<error>Please specify an app to update or "--all" to update all updatable apps"</error>');
return 1; return 1;

13
core/Command/Db/Migrations/GenerateCommand.php

@ -71,13 +71,10 @@ class {{classname}} extends SimpleMigrationStep {
} }
'; ';
protected Connection $connection;
protected IAppManager $appManager;
public function __construct(Connection $connection, IAppManager $appManager) {
$this->connection = $connection;
$this->appManager = $appManager;
public function __construct(
protected Connection $connection,
protected IAppManager $appManager,
) {
parent::__construct(); parent::__construct();
} }
@ -155,7 +152,7 @@ class {{classname}} extends SimpleMigrationStep {
*/ */
public function completeArgumentValues($argumentName, CompletionContext $context) { public function completeArgumentValues($argumentName, CompletionContext $context) {
if ($argumentName === 'app') { if ($argumentName === 'app') {
$allApps = \OC_App::getAllApps();
$allApps = $this->appManager->getAllAppsInAppsFolders();
return array_diff($allApps, \OC_App::getEnabledApps(true, true)); return array_diff($allApps, \OC_App::getEnabledApps(true, true));
} }

2
core/Command/Db/Migrations/GenerateMetadataCommand.php

@ -64,7 +64,7 @@ class GenerateMetadataCommand extends Command {
* @throws \Exception * @throws \Exception
*/ */
private function extractMigrationMetadataFromApps(): array { private function extractMigrationMetadataFromApps(): array {
$allApps = \OC_App::getAllApps();
$allApps = $this->appManager->getAllAppsInAppsFolders();
$metadata = []; $metadata = [];
foreach ($allApps as $appId) { foreach ($allApps as $appId) {
// We need to load app before being able to extract Migrations // We need to load app before being able to extract Migrations

9
core/Command/L10n/CreateJs.php

@ -11,6 +11,7 @@ namespace OC\Core\Command\L10n;
use DirectoryIterator; use DirectoryIterator;
use OCP\App\AppPathNotFoundException;
use OCP\App\IAppManager; use OCP\App\IAppManager;
use Stecman\Component\Symfony\Console\BashCompletion\Completion\CompletionAwareInterface; use Stecman\Component\Symfony\Console\BashCompletion\Completion\CompletionAwareInterface;
use Stecman\Component\Symfony\Console\BashCompletion\CompletionContext; use Stecman\Component\Symfony\Console\BashCompletion\CompletionContext;
@ -147,10 +148,14 @@ class CreateJs extends Command implements CompletionAwareInterface {
*/ */
public function completeArgumentValues($argumentName, CompletionContext $context) { public function completeArgumentValues($argumentName, CompletionContext $context) {
if ($argumentName === 'app') { if ($argumentName === 'app') {
return \OC_App::getAllApps();
return $this->appManager->getAllAppsInAppsFolders();
} elseif ($argumentName === 'lang') { } elseif ($argumentName === 'lang') {
$appName = $context->getWordAtIndex($context->getWordIndex() - 1); $appName = $context->getWordAtIndex($context->getWordIndex() - 1);
return $this->getAllLanguages(\OC_App::getAppPath($appName));
try {
return $this->getAllLanguages($this->appManager->getAppPath($appName));
} catch(AppPathNotFoundException) {
return [];
}
} }
return []; return [];
} }

40
lib/private/App/AppManager.php

@ -155,6 +155,37 @@ class AppManager implements IAppManager {
return array_keys($this->getInstalledAppsValues()); return array_keys($this->getInstalledAppsValues());
} }
/**
* Get a list of all apps in the apps folder
*
* @return list<string> an array of app names (string IDs)
*/
public function getAllAppsInAppsFolders(): array {
$apps = [];
foreach (\OC::$APPSROOTS as $apps_dir) {
if (!is_readable($apps_dir['path'])) {
$this->logger->warning('unable to read app folder : ' . $apps_dir['path'], ['app' => 'core']);
continue;
}
$dh = opendir($apps_dir['path']);
if (is_resource($dh)) {
while (($file = readdir($dh)) !== false) {
if (
$file[0] != '.' &&
is_dir($apps_dir['path'] . '/' . $file) &&
is_file($apps_dir['path'] . '/' . $file . '/appinfo/info.xml')
) {
$apps[] = $file;
}
}
}
}
return array_values(array_unique($apps));
}
/** /**
* List all apps enabled for a user * List all apps enabled for a user
* *
@ -647,11 +678,9 @@ class AppManager implements IAppManager {
/** /**
* Get the directory for the given app. * Get the directory for the given app.
* *
* @param string $appId
* @return string
* @throws AppPathNotFoundException if app folder can't be found * @throws AppPathNotFoundException if app folder can't be found
*/ */
public function getAppPath($appId) {
public function getAppPath(string $appId): string {
$appPath = \OC_App::getAppPath($appId); $appPath = \OC_App::getAppPath($appId);
if ($appPath === false) { if ($appPath === false) {
throw new AppPathNotFoundException('Could not find path for ' . $appId); throw new AppPathNotFoundException('Could not find path for ' . $appId);
@ -877,4 +906,9 @@ class AppManager implements IAppManager {
return false; return false;
} }
public function cleanAppId(string $app): string {
// FIXME should list allowed characters instead
return str_replace(['<', '>', '"', "'", '\0', '/', '\\', '..'], '', $app);
}
} }

30
lib/private/Authentication/TwoFactorAuth/ProviderLoader.php

@ -9,8 +9,7 @@ declare(strict_types=1);
namespace OC\Authentication\TwoFactorAuth; namespace OC\Authentication\TwoFactorAuth;
use Exception; use Exception;
use OC;
use OC_App;
use OC\AppFramework\Bootstrap\Coordinator;
use OCP\App\IAppManager; use OCP\App\IAppManager;
use OCP\AppFramework\QueryException; use OCP\AppFramework\QueryException;
use OCP\Authentication\TwoFactorAuth\IProvider; use OCP\Authentication\TwoFactorAuth\IProvider;
@ -19,15 +18,10 @@ use OCP\IUser;
class ProviderLoader { class ProviderLoader {
public const BACKUP_CODES_APP_ID = 'twofactor_backupcodes'; public const BACKUP_CODES_APP_ID = 'twofactor_backupcodes';
/** @var IAppManager */
private $appManager;
/** @var OC\AppFramework\Bootstrap\Coordinator */
private $coordinator;
public function __construct(IAppManager $appManager, OC\AppFramework\Bootstrap\Coordinator $coordinator) {
$this->appManager = $appManager;
$this->coordinator = $coordinator;
public function __construct(
private IAppManager $appManager,
private Coordinator $coordinator,
) {
} }
/** /**
@ -58,12 +52,12 @@ class ProviderLoader {
} }
} }
$registeredProviders = $this->coordinator->getRegistrationContext()->getTwoFactorProviders();
$registeredProviders = $this->coordinator->getRegistrationContext()?->getTwoFactorProviders() ?? [];
foreach ($registeredProviders as $provider) { foreach ($registeredProviders as $provider) {
try { try {
$this->loadTwoFactorApp($provider->getAppId()); $this->loadTwoFactorApp($provider->getAppId());
$provider = \OCP\Server::get($provider->getService());
$providers[$provider->getId()] = $provider;
$providerInstance = \OCP\Server::get($provider->getService());
$providers[$providerInstance->getId()] = $providerInstance;
} catch (QueryException $exc) { } catch (QueryException $exc) {
// Provider class can not be resolved // Provider class can not be resolved
throw new Exception('Could not load two-factor auth provider ' . $provider->getService()); throw new Exception('Could not load two-factor auth provider ' . $provider->getService());
@ -75,12 +69,10 @@ class ProviderLoader {
/** /**
* Load an app by ID if it has not been loaded yet * Load an app by ID if it has not been loaded yet
*
* @param string $appId
*/ */
protected function loadTwoFactorApp(string $appId) {
if (!OC_App::isAppLoaded($appId)) {
OC_App::loadApp($appId);
protected function loadTwoFactorApp(string $appId): void {
if (!$this->appManager->isAppLoaded($appId)) {
$this->appManager->loadApp($appId);
} }
} }
} }

4
lib/private/IntegrityCheck/Checker.php

@ -46,7 +46,7 @@ class Checker {
private ?IConfig $config, private ?IConfig $config,
private ?IAppConfig $appConfig, private ?IAppConfig $appConfig,
ICacheFactory $cacheFactory, ICacheFactory $cacheFactory,
private ?IAppManager $appManager,
private IAppManager $appManager,
private IMimeTypeDetector $mimeTypeDetector, private IMimeTypeDetector $mimeTypeDetector,
) { ) {
$this->cache = $cacheFactory->createDistributed(self::CACHE_KEY); $this->cache = $cacheFactory->createDistributed(self::CACHE_KEY);
@ -536,7 +536,7 @@ class Checker {
public function runInstanceVerification() { public function runInstanceVerification() {
$this->cleanResults(); $this->cleanResults();
$this->verifyCoreSignature(); $this->verifyCoreSignature();
$appIds = $this->appLocator->getAllApps();
$appIds = $this->appManager->getAllAppsInAppsFolders();
foreach ($appIds as $appId) { foreach ($appIds as $appId) {
// If an application is shipped a valid signature is required // If an application is shipped a valid signature is required
$isShipped = $this->appManager->isShipped($appId); $isShipped = $this->appManager->isShipped($appId);

9
lib/private/IntegrityCheck/Helpers/AppLocator.php

@ -30,13 +30,4 @@ class AppLocator {
} }
return $path; return $path;
} }
/**
* Providers \OC_App::getAllApps()
*
* @return array
*/
public function getAllApps(): array {
return \OC_App::getAllApps();
}
} }

2
lib/private/L10N/Factory.php

@ -81,7 +81,7 @@ class Factory implements IFactory {
*/ */
public function get($app, $lang = null, $locale = null) { public function get($app, $lang = null, $locale = null) {
return new LazyL10N(function () use ($app, $lang, $locale) { return new LazyL10N(function () use ($app, $lang, $locale) {
$app = \OC_App::cleanAppId($app);
$app = $this->appManager->cleanAppId($app);
if ($lang !== null) { if ($lang !== null) {
$lang = str_replace(['\0', '/', '\\', '..'], '', $lang); $lang = str_replace(['\0', '/', '\\', '..'], '', $lang);
} }

26
lib/private/Route/Router.php

@ -104,7 +104,7 @@ class Router implements IRouter {
*/ */
public function loadRoutes($app = null) { public function loadRoutes($app = null) {
if (is_string($app)) { if (is_string($app)) {
$app = \OC_App::cleanAppId($app);
$app = $this->appManager->cleanAppId($app);
} }
$requestedApp = $app; $requestedApp = $app;
@ -123,11 +123,15 @@ class Router implements IRouter {
if (isset($this->loadedApps[$app])) { if (isset($this->loadedApps[$app])) {
return; return;
} }
$appPath = \OC_App::getAppPath($app);
$file = $appPath . '/appinfo/routes.php';
if ($appPath !== false && file_exists($file)) {
$routingFiles = [$app => $file];
} else {
try {
$appPath = $this->appManager->getAppPath($app);
$file = $appPath . '/appinfo/routes.php';
if (file_exists($file)) {
$routingFiles = [$app => $file];
} else {
$routingFiles = [];
}
} catch (AppPathNotFoundException) {
$routingFiles = []; $routingFiles = [];
} }
@ -238,14 +242,14 @@ class Router implements IRouter {
// empty string / 'apps' / $app / rest of the route // empty string / 'apps' / $app / rest of the route
[, , $app,] = explode('/', $url, 4); [, , $app,] = explode('/', $url, 4);
$app = \OC_App::cleanAppId($app);
$app = $this->appManager->cleanAppId($app);
\OC::$REQUESTEDAPP = $app; \OC::$REQUESTEDAPP = $app;
$this->loadRoutes($app); $this->loadRoutes($app);
} elseif (str_starts_with($url, '/ocsapp/apps/')) { } elseif (str_starts_with($url, '/ocsapp/apps/')) {
// empty string / 'ocsapp' / 'apps' / $app / rest of the route // empty string / 'ocsapp' / 'apps' / $app / rest of the route
[, , , $app,] = explode('/', $url, 5); [, , , $app,] = explode('/', $url, 5);
$app = \OC_App::cleanAppId($app);
$app = $this->appManager->cleanAppId($app);
\OC::$REQUESTEDAPP = $app; \OC::$REQUESTEDAPP = $app;
$this->loadRoutes($app); $this->loadRoutes($app);
} elseif (str_starts_with($url, '/settings/')) { } elseif (str_starts_with($url, '/settings/')) {
@ -433,7 +437,11 @@ class Router implements IRouter {
$appControllerPath = __DIR__ . '/../../../core/Controller'; $appControllerPath = __DIR__ . '/../../../core/Controller';
$appNameSpace = 'OC\\Core'; $appNameSpace = 'OC\\Core';
} else { } else {
$appControllerPath = \OC_App::getAppPath($app) . '/lib/Controller';
try {
$appControllerPath = $this->appManager->getAppPath($app) . '/lib/Controller';
} catch (AppPathNotFoundException) {
return [];
}
$appNameSpace = App::buildAppNamespace($app); $appNameSpace = App::buildAppNamespace($app);
} }

2
lib/private/Server.php

@ -954,10 +954,10 @@ class Server extends ServerContainer implements IServerContainer {
if (\OC::$server->get(SystemConfig::class)->getValue('installed', false)) { if (\OC::$server->get(SystemConfig::class)->getValue('installed', false)) {
$config = $c->get(\OCP\IConfig::class); $config = $c->get(\OCP\IConfig::class);
$appConfig = $c->get(\OCP\IAppConfig::class); $appConfig = $c->get(\OCP\IAppConfig::class);
$appManager = $c->get(IAppManager::class);
} else { } else {
$config = $appConfig = $appManager = null; $config = $appConfig = $appManager = null;
} }
$appManager = $c->get(IAppManager::class);
return new Checker( return new Checker(
new EnvironmentHelper(), new EnvironmentHelper(),

31
lib/private/legacy/OC_App.php

@ -45,8 +45,7 @@ class OC_App {
* @psalm-taint-escape html * @psalm-taint-escape html
* @psalm-taint-escape has_quotes * @psalm-taint-escape has_quotes
* *
* @param string $app AppId that needs to be cleaned
* @return string
* @deprecated 31.0.0 use IAppManager::cleanAppId
*/ */
public static function cleanAppId(string $app): string { public static function cleanAppId(string $app): string {
return str_replace(['<', '>', '"', "'", '\0', '/', '\\', '..'], '', $app); return str_replace(['<', '>', '"', "'", '\0', '/', '\\', '..'], '', $app);
@ -469,30 +468,10 @@ class OC_App {
* get a list of all apps in the apps folder * get a list of all apps in the apps folder
* *
* @return string[] an array of app names (string IDs) * @return string[] an array of app names (string IDs)
* @todo: change the name of this method to getInstalledApps, which is more accurate
* @deprecated 31.0.0 Use IAppManager::getAllAppsInAppsFolders instead
*/ */
public static function getAllApps(): array { public static function getAllApps(): array {
$apps = [];
foreach (OC::$APPSROOTS as $apps_dir) {
if (!is_readable($apps_dir['path'])) {
\OCP\Server::get(LoggerInterface::class)->warning('unable to read app folder : ' . $apps_dir['path'], ['app' => 'core']);
continue;
}
$dh = opendir($apps_dir['path']);
if (is_resource($dh)) {
while (($file = readdir($dh)) !== false) {
if ($file[0] != '.' and is_dir($apps_dir['path'] . '/' . $file) and is_file($apps_dir['path'] . '/' . $file . '/appinfo/info.xml')) {
$apps[] = $file;
}
}
}
}
$apps = array_unique($apps);
return $apps;
return \OCP\Server::get(IAppManager::class)->getAllAppsInAppsFolders();
} }
/** /**
@ -513,9 +492,9 @@ class OC_App {
* @return array * @return array
*/ */
public function listAllApps(): array { public function listAllApps(): array {
$installedApps = OC_App::getAllApps();
$appManager = \OC::$server->getAppManager(); $appManager = \OC::$server->getAppManager();
$installedApps = $appManager->getAllAppsInAppsFolders();
//we don't want to show configuration for these //we don't want to show configuration for these
$blacklist = $appManager->getAlwaysEnabledApps(); $blacklist = $appManager->getAlwaysEnabledApps();
$appList = []; $appList = [];

24
lib/public/App/IAppManager.php

@ -141,12 +141,10 @@ interface IAppManager {
/** /**
* Get the directory for the given app. * Get the directory for the given app.
* *
* @param string $appId
* @return string
* @since 11.0.0 * @since 11.0.0
* @throws AppPathNotFoundException * @throws AppPathNotFoundException
*/ */
public function getAppPath($appId);
public function getAppPath(string $appId): string;
/** /**
* Get the web path for the given app. * Get the web path for the given app.
@ -282,4 +280,24 @@ interface IAppManager {
* @since 30.0.0 * @since 30.0.0
*/ */
public function isBackendRequired(string $backend): bool; public function isBackendRequired(string $backend): bool;
/**
* Clean the appId from forbidden characters
*
* @psalm-taint-escape file
* @psalm-taint-escape include
* @psalm-taint-escape html
* @psalm-taint-escape has_quotes
*
* @since 31.0.0
*/
public function cleanAppId(string $app): string;
/**
* Get a list of all apps in the apps folder
*
* @return list<string> an array of app names (string IDs)
* @since 31.0.0
*/
public function getAllAppsInAppsFolders(): array;
} }

19
remote.php

@ -8,6 +8,7 @@
require_once __DIR__ . '/lib/versioncheck.php'; require_once __DIR__ . '/lib/versioncheck.php';
use OCA\DAV\Connector\Sabre\ExceptionLoggerPlugin; use OCA\DAV\Connector\Sabre\ExceptionLoggerPlugin;
use OCP\App\IAppManager;
use Psr\Log\LoggerInterface; use Psr\Log\LoggerInterface;
use Sabre\DAV\Exception\ServiceUnavailable; use Sabre\DAV\Exception\ServiceUnavailable;
use Sabre\DAV\Server; use Sabre\DAV\Server;
@ -20,10 +21,7 @@ use Sabre\DAV\Server;
class RemoteException extends \Exception { class RemoteException extends \Exception {
} }
/**
* @param Exception|Error $e
*/
function handleException($e) {
function handleException(Exception|Error $e): void {
try { try {
$request = \OC::$server->getRequest(); $request = \OC::$server->getRequest();
// in case the request content type is text/xml - we assume it's a WebDAV request // in case the request content type is text/xml - we assume it's a WebDAV request
@ -126,20 +124,21 @@ try {
// Load all required applications // Load all required applications
\OC::$REQUESTEDAPP = $app; \OC::$REQUESTEDAPP = $app;
OC_App::loadApps(['authentication']);
OC_App::loadApps(['extended_authentication']);
OC_App::loadApps(['filesystem', 'logging']);
$appManager = \OCP\Server::get(IAppManager::class);
$appManager->loadApps(['authentication']);
$appManager->loadApps(['extended_authentication']);
$appManager->loadApps(['filesystem', 'logging']);
switch ($app) { switch ($app) {
case 'core': case 'core':
$file = OC::$SERVERROOT .'/'. $file; $file = OC::$SERVERROOT .'/'. $file;
break; break;
default: default:
if (!\OC::$server->getAppManager()->isInstalled($app)) {
if (!$appManager->isInstalled($app)) {
throw new RemoteException('App not installed: ' . $app); throw new RemoteException('App not installed: ' . $app);
} }
OC_App::loadApp($app);
$file = OC_App::getAppPath($app) .'/'. $parts[1];
$appManager->loadApp($app);
$file = $appManager->getAppPath($app) .'/'. ($parts[1] ?? '');
break; break;
} }
$baseuri = OC::$WEBROOT . '/remote.php/'.$service.'/'; $baseuri = OC::$WEBROOT . '/remote.php/'.$service.'/';

11
tests/apps.php

@ -44,10 +44,15 @@ function getSubclasses($parentClassName): array {
} }
$apps = OC_App::getEnabledApps(); $apps = OC_App::getEnabledApps();
$appManager = \OCP\Server::get(\OCP\App\IAppManager::class);
foreach ($apps as $app) { foreach ($apps as $app) {
$dir = OC_App::getAppPath($app);
if (is_dir($dir . '/tests')) {
loadDirectory($dir . '/tests');
try {
$dir = $appManager->getAppPath($app);
if (is_dir($dir . '/tests')) {
loadDirectory($dir . '/tests');
}
} catch (\OCP\App\AppPathNotFoundException) {
/* ignore */
} }
} }

4
tests/lib/App/AppManagerTest.php

@ -341,7 +341,7 @@ class AppManagerTest extends TestCase {
$manager->expects($this->once()) $manager->expects($this->once())
->method('getAppPath') ->method('getAppPath')
->with('test') ->with('test')
->willReturn(null);
->willReturn('');
$manager->expects($this->once()) $manager->expects($this->once())
->method('getAppInfo') ->method('getAppInfo')
@ -402,7 +402,7 @@ class AppManagerTest extends TestCase {
$manager->expects($this->once()) $manager->expects($this->once())
->method('getAppPath') ->method('getAppPath')
->with('test') ->with('test')
->willReturn(null);
->willReturn('');
$manager->expects($this->once()) $manager->expects($this->once())
->method('getAppInfo') ->method('getAppInfo')

12
tests/lib/InfoXmlTest.php

@ -7,6 +7,7 @@
namespace Test; namespace Test;
use OCP\App\IAppManager; use OCP\App\IAppManager;
use OCP\Server;
/** /**
* Class InfoXmlTest * Class InfoXmlTest
@ -15,6 +16,13 @@ use OCP\App\IAppManager;
* @package Test * @package Test
*/ */
class InfoXmlTest extends TestCase { class InfoXmlTest extends TestCase {
private IAppManager $appManager;
protected function setUp(): void {
parent::setUp();
$this->appManager = Server::get(IAppManager::class);
}
public function dataApps() { public function dataApps() {
return [ return [
['admin_audit'], ['admin_audit'],
@ -45,8 +53,8 @@ class InfoXmlTest extends TestCase {
* @param string $app * @param string $app
*/ */
public function testClasses($app) { public function testClasses($app) {
$appInfo = \OCP\Server::get(IAppManager::class)->getAppInfo($app);
$appPath = \OC_App::getAppPath($app);
$appInfo = $this->appManager->getAppInfo($app);
$appPath = $this->appManager->getAppPath($app);
\OC_App::registerAutoloading($app, $appPath); \OC_App::registerAutoloading($app, $appPath);
//Add the appcontainer //Add the appcontainer

4
tests/lib/IntegrityCheck/CheckerTest.php

@ -1030,9 +1030,9 @@ class CheckerTest extends TestCase {
$this->checker $this->checker
->expects($this->once()) ->expects($this->once())
->method('verifyCoreSignature'); ->method('verifyCoreSignature');
$this->appLocator
$this->appManager
->expects($this->once()) ->expects($this->once())
->method('getAllApps')
->method('getAllAppsInAppsFolders')
->willReturn([ ->willReturn([
'files', 'files',
'calendar', 'calendar',

4
tests/lib/IntegrityCheck/Helpers/AppLocatorTest.php

@ -30,8 +30,4 @@ class AppLocatorTest extends TestCase {
$this->locator->getAppPath('aTotallyNotExistingApp'); $this->locator->getAppPath('aTotallyNotExistingApp');
} }
public function testGetAllApps() {
$this->assertSame(\OC_App::getAllApps(), $this->locator->getAllApps());
}
} }

14
tests/lib/Route/RouterTest.php

@ -13,6 +13,7 @@ use OCP\App\IAppManager;
use OCP\Diagnostics\IEventLogger; use OCP\Diagnostics\IEventLogger;
use OCP\IConfig; use OCP\IConfig;
use OCP\IRequest; use OCP\IRequest;
use PHPUnit\Framework\MockObject\MockObject;
use Psr\Container\ContainerInterface; use Psr\Container\ContainerInterface;
use Psr\Log\LoggerInterface; use Psr\Log\LoggerInterface;
use Test\TestCase; use Test\TestCase;
@ -26,6 +27,8 @@ use Test\TestCase;
*/ */
class RouterTest extends TestCase { class RouterTest extends TestCase {
private Router $router; private Router $router;
private IAppManager&MockObject $appManager;
protected function setUp(): void { protected function setUp(): void {
parent::setUp(); parent::setUp();
/** @var LoggerInterface $logger */ /** @var LoggerInterface $logger */
@ -36,13 +39,16 @@ class RouterTest extends TestCase {
$this->fail('Unexpected info log: '.(string)($data['exception'] ?? $message)); $this->fail('Unexpected info log: '.(string)($data['exception'] ?? $message));
} }
); );
$this->appManager = $this->createMock(IAppManager::class);
$this->router = new Router( $this->router = new Router(
$logger, $logger,
$this->createMock(IRequest::class), $this->createMock(IRequest::class),
$this->createMock(IConfig::class), $this->createMock(IConfig::class),
$this->createMock(IEventLogger::class), $this->createMock(IEventLogger::class),
$this->createMock(ContainerInterface::class), $this->createMock(ContainerInterface::class),
$this->createMock(IAppManager::class),
$this->appManager,
); );
} }
@ -51,6 +57,12 @@ class RouterTest extends TestCase {
} }
public function testGenerateConsecutively(): void { public function testGenerateConsecutively(): void {
$this->appManager->expects(self::atLeastOnce())
->method('cleanAppId')
->willReturnArgument(0);
$this->appManager->expects(self::atLeastOnce())
->method('getAppPath')
->willReturnCallback(fn (string $appid): string => \OC::$SERVERROOT . '/apps/' . $appid);
$this->assertEquals('/index.php/apps/files/', $this->router->generate('files.view.index')); $this->assertEquals('/index.php/apps/files/', $this->router->generate('files.view.index'));

Loading…
Cancel
Save